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修饰符的变量也不能持有对象.
缺点:和__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;
然而下面的这种情况又是怎么回事?
NSError *err = nil;默认是 __strong修饰符.而方法的参数声明是__autoreleasing修饰符. NSError *err = nil; Bool result = [obj performOperationWithError:&err];

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

实际上,是编译器自动将上述源代码做了转换变成:
NSError __strong *error = nil; NSError __autoreleasing *tmp = error; BOOL result = [obj performOperationWithError:&tmp]; error = tmp;

属性关键字:
属性关键字有:
1.assign 对应__unsafe_unretained修饰符
2.copy 对应__strong修饰符(但是赋值的是被复制的对象) MRC就有
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;

推荐阅读更多精彩内容