在定义属性的时候,需要在括号内说明该属性的特性。属性的特性决定了属性在原子性、存取方法以及内存管理3个方面的特性。目前,常用的属性关键字有8个,分别为nonatomic,atomic,readonly,readwrite,strong,weak,assign,copy。
1.原子性(atomic,nonatomic)
(1)atomic(默认):atomic意为操作是原子的,意味着只有一个线程访问实例变量。atomic是线程安全的,至少在当前的存取器上是安全的。它是一个默认的特性,但是很少使用,因为比较影响效率。当一个变量声明为atomic时,意味着在多线程中只能有一个线程能对它进行访问;当一个变量声明为atomic时,该变量为线程安全型,但是会影响访问速度;当一个变量声明为atomic时,在非ARC编译环境下,需要设置访问锁来保证该变量进行正确的getter/setter。
(2)nonatomic(常用):nonatomic跟atomic刚好相反,表示非原子的,可以被多个线程访问。它的效率比atomic快,但不能保证在多线程环境下的安全性,在单线程和明确只有一个线程访问的情况下广泛使用。当一个变量声明为nonatomic时,意味着多个线程可以同时对其进行访问;当一个变量声明为nonatomic时,它是非线程安全型,访问速度快;当一个变量声明为nonatomic时,当两个不同的线程对其访问时,容易失控。
2.存取方法(readwrite,readonly)
readwrite(默认):表示该属性同时拥有setter和getter,即该属性既可以读,也可以写。
readonly:表示只有getter没有setter,即该属性只能读取,不能更新或写入。
有时,在声明属性时,可以指定存取方法的自定义名称,通常在BOOL类型属性的getter方法会采用这种方式,例如:UIView的hidden和opaque属性的getter方法名为:isHidden和isOpaque。
3.内存管理(strong,weak,assign,copy)
由于苹果自iOS5开始引入了ARC(Automatic Reference Counting),且经过几年发展ARC已成为主流,因此在管理属性有关内存管理的特性时,都是假设在ARC环境下的,不再考虑MRC的情况。在ARC环境下,与内存管理相关的几个关键词含义如下。
strong(ARC中为默认选项):强引用,表示实例变量对传入的对象要有所有权关系,引用计数加1。
weak:弱引用,在setter方法中,对传入的对象不进行引用计数加1的操作。简单来说,就是对传入的对象没有所有权,当该对象引用计数变为0时,即该对象被释放后,用weak声明的实例变量指向nil。
assign:简单赋值,不改变引用计数,适用简单数据类型。如int、float、double、NSInteger,CGFloat等。
copy:在内存中保留一份传入值的复制,而不是值自身的情况,即把传入的原对象完整地复制到另外一个新的内存区,当副本改变时,原对象不会发生改变。同样,当原对象发生改变时,其副本也不会发生改变,因为原对象和副本对象存储在独立的两个内存区域中。copy对传入的原对象的副本有所有权关系,而并非原对象本身。
示例代码
我们可以通过一段示例代码来观察一下属性关键字对于属性存储状态的影响。创建一个ClassA类,并添加3个NSMutableString类型的属性,但关键字不同,如下所示。
在main()中,添加如下代码,通过属性对应的实例变量的内存地址,可以对strong/weak/copy有所理解。
其中strong/weak修饰的属性的指针直接指向string的地址,而copy会创建一个新的副本,故内存地址会不一样。
运行结果如下所示: