理解 属性 的概念
属性会自动生成存取方法, 可以利用点语法调用,
若不想编译器自动合成存取方法, 可以自己实现, 还有另外一种方法, 就是使用 @dynamic 关键字, 它会告诉编译器, 不要自动创建实现属性所用的实例变量 和 存取方法, 而且,在编译访问属性的代码时, 即使编译器发现没有定义存取方法, 也不会报错, 它相信这些方法会在运行期找到, 使用方法是, 在 头文件中(.h 文件) 定义属性, 在 实现文件 (.m 文件) 使用 @dynamic 属性名;
定义了属性之后, 可以在类的 实现代码 (即.m 文件)中通过 @synthesize 语法来指定实例变量的名字, 而不再使用默认的名字, 但是一般不推荐使用此方法, 因为不方便阅读
// 属性特质
属性可以拥有的特质分为四类:
定义: 在 并发编程中. 如果其操作具备 整体性, 也就是说, 系统 其它部分 无法观察 到 其中间步骤所生成的 临时结果, 而只能看到 操作前 和 操作后 的结果, 那么该操作就是 '原子的'(atomic), 或者说, 该操作具备 '原子性'(atomicty)
在默认情况下, 有编译器所合成的方法会通过锁定机制确保其原子性 (atomicity), 如果属性具备nonatomic 特质, 则不使用同步锁, 请注意, 尽管没有名为 'atomicity' 的特质, 但是如果属性不具备 nonatomic 特质, 那它就是 '原子的', 也可以在属性的特质中写明这一点, 编译器不会报错
atomic 与 nonatomic 的区别
具备 atomic 特质的获取方法会通过锁定机制来确保其操作的原子性. 也就是说, 如果两个线程同时读取一个属性,那么无论何时, 总能得到有效的属性值 (当线程A进行写操作,这时其他线程的读或者写操作会因为等该操作而等待。当A线程的写操作结束后,B线程进行写操作,所有这些不同线程上的操作都将依次顺序执行——也就是说,如果一个线程正在执行 getter/setter,其他线程就得等待。)
使用 nonatomic , 如果其中一个线程正在改变其属性值的时候, 另外一个线程也许会突然闯入, 把尚未修改好的属性值读取出来, 这种情况下, 线程读取到的属性值 肯定 不一致
一般 iOS 程序中, 所有属性都声明为 nonatomic , 这样是因为在 iOS 中使用同步锁的开销比较大, 会带来性能问题, 一般情况下也不会要求属性必须是原子的, 而且即使设置了 atomic 也不能保证绝对的线程安全
读/写 权限
具备 readwrite (可读可写)特质的属性拥有 获取方法 (getter) 与 设置方法(setter),
具备 readonly (只读) 特质的属性仅拥有 获取方法
内存管理语义
下面这一组特质仅会影响 '设置方法', 编译器在合成存取代码时, 要证据此特质来决定所生成的代码, 如果自己编写存取方法, 那么就必须同有关属性所具备的特质相符 .
assign 设置方法 只会执行 纯量类型 (如 CGFloat 或 NSInteger) 的简单赋值操作
strong 此特质表明该属性定义了一种 '拥有关系', 为这种属性设置新值时, 设置方法会先保留新值, 并释放旧值, 然后在将新值设置上去
weak 此特质表明了一种 '非拥有关系', 为这种属性设置新值时, 设置方法既不保留新值, 也不释放旧值, 然而在属性所指的对象遭到摧毁时, 属性值也会清空
copy 此特质所表达的所述关系与 strong 类似, 然而设置方法并不保留新值, 而是将其拷贝,( 当属性类型为 NSString * 时, 经常用此特质来保护其封装性, 因为传递给设置方法的新值有可能指向一个 NSMubleString 类的实例, 这个类是 NSString 的子类, 表示一种可以修改其值的字符串, 此时若是不拷贝字符串,那么设置完属性之后, 字符串的值就可能会在对象不知情的情况下造人更改), 只要实现属性所用的对象是 '可变的', 就应该在设置新属性值时拷贝一份.
方法名
可通过如下特质来指定存取方法的方法名
getter=<name> 指定 '获取方法' 的方法名, 如果某属性是 Boolean 型, 而你想为其获取方法加上 'is' 前缀. 那么就可以用这个方法来指定
例如: @property (nonatomic, getter-isOn) BOOL on;
setter=<name> 指定 '设置方法'的方法名
通过上述特质, 可以微调由编译器所合成的存取方法, 不过需要注意的是, 若是自己来实现这些存取方法, 那么应该保证其具备相关属性所声明的特质.
如果想在其他方法里设置属性值, 那么同样要遵循属性定义中所宣称的语义, 例如:一个类新增了一个初始化方法
@interface EOCPerson : NSManagedObject
@property (copy) NSString * firstName;
@property (copy) NSString * lastName;
- (id)initWithFirstName:(NSString*)firstName lastName:(NSString *)lastName;
@end
在实现这个自定义的初始化方法时, 一定要遵循属性定义中宣城的 'copy' 语义, 因为 '属性定义' 就相当于 '类' 和 '待设置的属性值' 之间所达成的 契约
初始化的方法可以这么写:
- (id)initWithFirstName:(NSString*)firstName lastName:(NSString *)lastName{
if (self == [super init]){
_firstName = [firstName copy];
_lastName = [lastName copy];
}
}
总结:
可以通过 '特质' 来指定存储数据所需的正确语义
在设置属性所对应的实例变量时, 一定要遵循从该属性所声明的语义