前言
今天和大家一起来探讨一下OC中的property(属性)这一概念.
OC用property来封装对象里的数据,通过property,可以让编译器创建一个实例变量来保存数据,并且创建对应的一对存取器方法,来进行对该数据的访问.
有了property之后,我们便可以方便的使用点语法来访问对象中所封装的数据了,实在是方便的很.
不过property还是有诸多细节值得我们讨论的,下面,我们一一对其探讨.
property的实质
@property语法,其实是一种语法糖,也就是说,它其实就是一个下划线实例变量和一对存储器方法的集合
@property NSString *name;
上面这种语法和下面这种写法是等效的:
{
NSString * _name;
}
-(NSString *)name
{
return _name;
}
-(void)setName:(NSString *)name
{
_name=name;
}
我们再编写@property属性的时候编译器会在编译期为这些属性生成访问这些属性所需的存取器方法,这个过程叫做自动合成(autosynthesis)
除了存取器方法之外,编译器也会自动生成对应的实例变量名字--在property名字之前加上'_'来作为新生成的实例变量的名字.
property的特质
在这里需要注意的就是,属性后的特质语义会影响编译器所生成的存取方法.这些特质语义分成四个方面:
原子性
原子性特质语义分为:atomic和nonatomic两种,如果不对原子性特质显示说明,则系统默认为atomic保留其原子性,使用同步锁.
虽然拥有这两种原子性的语义,但是,大家能发现,我们一般是使用nonatomic来标明属性的,这是因为atomic的使用在iOS开发中会带来性能瓶颈,而且,并不一定真的能保证线程安全,如果要线程安全,需要更深层次的同步锁.
在OSX开发中atomic通常不会带来性能瓶颈.
读写权限
读写特质分为:readwrite和readonly两种,默认为readwrite如果该属性说明为readonly则编译器不会为该属性生成set方法.
内存管理
内存管理相关有以下几种语义:
assign:此语义表明的属性,其设置方法只会对纯量类型的实例变量进行简单的赋值操作,例如:(NSInteger)类型,所以此种语义并不适合OC对象.
strong:此语义为该OC对象属性定义了一种"拥有关系",这种语义会先保留新值,然后释放旧值,最后设置为新值.
- weak:weak语义为该OC对象定义了一种"非拥有关系",这种语义并不会拥有该对象,而仅仅是简单的指向该对象,值得,当拥有该对象的其它对象都对其释放后,它便会销毁,而用weak语义指向它的属性会被置为nil,值得一提的是,如何变为nil而并不是野指针,这项操作涉及到了运行时相关的一些内容,我们日后再详细讨论.
- unsafe_unretained:这种语义很不常用,原因是它和weak类似,表示一种非拥有关系,但是和weak不同的是,当所指向的对象被销毁时,它的值并不会置为nil,所以是不安全的,一般并不适用.
-
copy: 这种语义表示的也是一种拥有关系,和strong不同的是,该语义并不是保留新值,而是copy拷贝一份新的值,常常用来在NSString类型的属性时适用,因为:很有可能该字符串实际是指向NSMutableString类型的实例,这一点涉及到类族在foundation框架中的大量适用,我们以后会详细,说明,除此之外,所有可变类型的属性,都应该在设置值时copy一份.
当我们重写被赋予了copy语义的属性的set方法时以及对成员变量直接赋值时,我们应该遵守copy语义,应该如下写:
-(void)setName:(NSString *)name { _name=[name copy]; }