OC所有权修饰符与属性关键字

ARC有效时,id类型和对象类型必须附加所有权修饰符,所有权修饰符一共4种:

  1. __strong修饰符.(强引用会持有对象)
  2. __weak修饰符.(弱引用不会持有对象)
  3. __unsafe_unretained修饰符 (不会持有对象,有悬挂指针的风险)
  4. __autoreleasing修饰符 (对象会被注册到自动释放池里)

其中__strong修饰符是id类型和对象类型默认的所有权修饰符.
使用__strong,__weak,__autoreleasing修饰符的自动变量,会被初始化为nil.

何为内存泄露?
内存泄露指的是应当废弃的对象在超出其生存周期后继续存在.
仅使用__strong修饰符,会导致循环引用的发生.循环引用有两种:

  1. 相互循环引用,即两个对象间的相互强引用.
  2. 自引用导致的循环引用.

__weak修饰符
适用范围:只能修饰对象类型,不能用于基本数据类型.
__weak修饰符优点:
1.可以避免循环引用导致的内存泄露.因为弱引用不会持有对象.
2.在持有某对象的弱引用时,若该对象被废弃,则此弱引用将自动失效且指针变量将被赋值为nil.
__weak修饰符只能用于iOS5及以上,在iOS4只能使用__unsafe_unretained代替.不过现在已经到了最低适配版本iOS7了,所以可以放心使用__weak修饰符.
__weak修饰符缺点:
1.正因为附有__weak修饰符的指针变量,当它指向的对象被销毁时,系统会将该对象所有的__weak指针都置为nil,所以效率相比__unsafe_unretained修饰符要低.有时候优点也是缺点.就看你如何掌握好"度".
2.不能用于基本数据类型.

__unsafe_unretained修饰符
适用范围:可用于基本数据类型也可用于对象类型.不过一般使用 __unsafe_unretained修饰对象类型,如果是基本类型则Xcode会有警告。
附有__unsafe_unretained修饰符的变量也不能持有对象.
缺点:和__weak修饰符不同的是如果带__unsafe_unretained修饰符的变量指向的对象被废弃了那么该指针变量的值不会被置为nil,依然还是以前的值.但它已经是野指针了,再次访问将会崩溃,虽然不是每次都崩.

__autoreleasing修饰符
ARC有效时,将对象赋值给附有__autoreleasing修饰符的指针变量等价于在ARC无效时调用对象的autorelease方法,即将对象注册到autoreleasepool.
但是,显式地附加__autoreleasing修饰符同显式地__strong修饰符一样罕见.因为编译器会帮我们添加.
比如使用alloc.../开头以外的方法来取得的对象是已经被注册到了autoreleasepool里的(虽然ARC下该返回的对象不一定真的注册到autoreleasepool里,这里暂且这么理解).这是由于编译器会检查方法名是否以alloc.../开始,如果不是则自动将作为返回值的对象注册到autoreleasepool.
比如ARC有效时,下面的方法:

+ (id)array

{

   id obj = [[NSMutableArray alloc] init];

   return obj;

}

由于没有显式的指定所有权修饰符,所以 id obj等同于id __strong obj.由于return使得对象变量超出其作用域,所以该强引用指针变量指向的对象将被释放,但该对象作为函数的返回值,编译器会自动将其注册到autoreleasepool.这里也都没有使用显式地附加__autoreleasing修饰符.
另外一种可以不需要显式的使用__autoreleasing修饰符的情况就是:id的指针或对象的指针(也就是双重指针)在没有显式指定时会被附加上__autoreleasing修饰符.
最常见的例子就是,获取错误NSError时.
NSError *err = nil; Bool result = [obj performOperationWithError:&err];
该方法的声明为:
- (BOOL)performOperationWithError:(NSError **)error;
上述方法声明是等同于
- (BOOL)performOperationWithError:(NSError * __autoreleasing *)error;
这里再说一次:作为alloc/new/copy/mutableCopy方法返回值取得的对象是自己生成并持有的,其他情况下便是取得非自己生成并持有的对象.因此,使用附有__autoreleasing修饰符的变量作为对象取得参数,与除alloc/new/copy/mutableCopy外其他方法的返回值取得对象完全一样,都会注册到autoreleasepool,并取得非自己生成并持有的对象.

最后,赋值给对象指针时,所有权修饰符必须一致.
因此下面的源代码会产生编译器错误:

NSError *err = nil;
NSError **p = &err;

需要改为:

NSError *err = nil;
NSError * __strong *p = &err; //这里并不会导致对象的引用计数+1

//比如
NSObject * __strong *p1 = NULL;
{
      NSObject *obj = MATBaseViewController.new;
      p1 = &obj; //并不会导致obj引用计数+1
}
NSLog(@"obj=%@", *p1); //obj=(null)

然而下面的这种情况又是怎么回事?

NSError *err = nil;默认是 __strong修饰符.而方法的参数声明是__autoreleasing修饰符.
Bool result = [obj performOperationWithError:&err];

该方法的声明为:
- (BOOL)performOperationWithError:(NSError **)error;

实际上,是编译器自动将上述源代码做了转换变成:

NSError __strong *error = nil;
NSError __autoreleasing *tmp = error;
BOOL result = [obj performOperationWithError:&tmp]; //这里tmp的所有权修饰符就和参数的一致了
error = tmp;

属性关键字:
属性关键字有:
1.assign 对应__unsafe_unretained修饰符,和unsafe_unretained几乎没有区别,可以用于对象类型也可以用于基本类型的属性声名
由于assign不是所有权修饰符所以没有 __assign NSObject *_obj;这样的写法。正确的写法为__unsafe_unretained NSObject *_obj;
2.copy 和MRC里的一样,实际会调用对象的copy方法,所以要求对象需要实现NSCoping协议。
如果有重写copy属性的setter方法,则在赋值时需要调用copy方法,而不是简单的赋值。

- (void)setAcat:(ARCCat *)acat {
    _acat = [acat copy];
}

3.retain 对应__strong修饰符 在ARC模式下,依然可以使用.
4.strong 默认 对应__strong修饰符
5.unsafe_unretained 对应__unsafe_unretained修饰符
6.weak 对应__weak修饰符
7.atomic 默认
8.nonatomic
9.readonly
10.readwrite 默认
需要注意的是属性的关键字需要和它的实例变量的修饰符一致(当你不使用系统帮你生成的实例变量时)

@interface ViewController ()

{

    __autoreleasing NSString *_ttrsr;

}

@property (nonatomic, strong) NSString *ttrsr;

@end

这样写会报错.
需要改为__strong NSString *_ttrsr;NSString *_ttrsr;

参考
Transitioning to ARC Release Notes

Objective-C Runtime Programming Guide--Declared Properties

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容