iOS-Objective-C的属性修饰符

  在苹果引入了ARC之后,属性的也相对应的增加了一些修饰符。所以这里我们分别对MRC和ARC两种情况下的属性修饰符进行学习。其实主要的区别集中在对对象生命周期进行描述的属性修饰符会有所区别而已。

存取类型

任何属性都可以声明为readwrite或readonly,且默认设置为readwrite。

  • readwrite:程序自动创建setter/getter方法。
  • readonly:程序之创建getter方法。

此外,还可以自定义setter/getter方法。

@propery(setter=setId,getter=getId) int id;

原子性

  • atomic:生成的setter/getter操作为原子性的操作,执行性能较低(系统默认)。
  • noatomic:生成的setter/getter操作为非原子性的操作,执行性能较高。一般推荐手动设置为该属性。

生命周期管理

首先我们先区分一下在MRC和ARC下的属性都有使用哪些关于生命周期管理的修饰符。

MRC: assign, copy, retain
ARC: strong, weak, unsafe_unretained, copy

修饰符说明

然后我们对于每种属性修饰符做进一步说明,然后对他们之间的对比也进行说明。

assign: 简单赋值,不更改引用计数。一般用于基础类型的数据(NSInteger)和C语言类型数据(int,float,double,char,bool)。其在MRC下是默认值。

copy: 会拷贝传入的对象(即创建一个引用计数为1的新对象,但是内容与传入对象相同),并把新对象赋值给实例变量。常用与NSString,NSArray,NSDictionary,NSSet等。

retain: 释放旧对象,并使传入的新对象引用计数+1。此属性只能用于NSObject及其子类,而不能用于Core Foundation(因为其没有使用引用计数,需要另外使用CFRetain和CFRelease爱进行CF的内存管理)。

ARC加入的属性修饰符如下。

strong: 强引用,类似于retain。要求保留传入的对象,并放弃原有对象。一个对象只要被至少一个强引用指向,则其不会被释放,而当没有强引用指向时则会被释放。其在ARC下是对象类型的默认值。

weak: 弱引用,要求不保留传入的属性(既不会使传入的对象引用计数+1)。类似于assign,但与assign不同的是,当它们指向的对象被释放后,weak会被自动置为nil,而assign则不会,所以assign会导致“野指针”的出现,weak可以避免悬空指针。

unsafe_unretained: 其实质等同于assign。与weak的区别就是指向的对象如果被释放,其不会被置为nil,而导致悬空指针的出现。它是ARC模式下非对象属性的默认值。

所以综上所述,属性的默认值主要有以下情况。

  • MRC:(atomic, readwrite, assign)
  • ARC下对象类型属性:(atomic, readwrite, strong)
  • ARC下非对象类型:(atomic, readwrite, unsafe_unretained)

其他对比

1) copy/retain
答:copy会拷贝创建一个新的对象,并使得它的引用计数为1。retain则是Release旧值,retain新值,其本质是指针复制(浅复制),引用计数加1,而不会导致内容被复制。
如:一个NSString对象,内存地址为:0x1111,内容为@“Hello”。
(1)copy到另外一个NSString后,地址为0x2222,内容相同(新建一个内容,内容拷贝),新的对象引用计数为1,旧的对象内容没有改变,引用计数-1。
(2)retain到另外一个NSString后,地址相同(新建一个指针,指针拷贝),内容相同,对象的引用计数+1。

2)assign/retain(MRC情况下)
答:assign只是简单的赋值,如果它引用的对象被释放了,则会造成悬空指针的出现,此时再通过该引用访问对象则会导致程序crash。retain则是在引用计数的基础上,对对象引用计数+1,以获取对象的拥有权,这样只有当对象的引用计数为0时才会被释放(既没有别的引用指向它),这样可以避免访问一个被释放的对象。

3)assign/weak(ARC情况下,因为assign类似于unsafe_unretained,所以也可以说是weak和unsafe_unretained的区别)
答:assign不同的是,当它们指向的对象被释放后,weak会被自动置为nil,而assign则不会,所以assign会导致“野指针”的出现。

扩展

  上面那些属性描述符都是针对类中定义的属性而言的,实际上对于局部变量也有类似的关键字来修饰变量,常用主要有__strong,__weak,__unsafe_unretained, __autoreleasing

__string: 是默认引用类型的关键字。
__weak: 声明一个可以自动置nil的弱引用。
__unsafe_unretained: 弱引用,但是当指向对象被释放时,不会被置nil。所以会导致野指针的出现。
__autoreleasing:用来修饰一个函数的参数,这个参数会在函数返回的时候被自动释放。

对象的拷贝

  有些类有两个版本:一个是可修改的,另一个是不可修改的。无论是哪个版本,copy方法都会返回不可修改的版本。例如,NSMutableString的copy方法会返回NSString实例。如果要拷贝可修改的对象,就要使用mutableCopy。

参考资料

  1. Objective-C编程(第2版)
  2. Objective-C——retain/copy/assign/atomic/nonatomic/strong/weak/readonly/readwrite详解
  3. IOS ARC关键字说明