ARC有效时,id类型和对象类型必须附加所有权修饰符,所有权修饰符一共4种:
-
__strong
修饰符.(强引用会持有对象) -
__weak
修饰符.(弱引用不会持有对象) -
__unsafe_unretained
修饰符 (不会持有对象,有悬挂指针的风险) -
__autoreleasing
修饰符 (对象会被注册到自动释放池里)
其中__strong
修饰符是id类型和对象类型默认的所有权修饰符.
使用__strong,__weak,__autoreleasing
修饰符的自动变量,会被初始化为nil.
何为内存泄露?
内存泄露指的是应当废弃的对象在超出其生存周期后继续存在.
仅使用__strong
修饰符,会导致循环引用的发生.循环引用有两种:
- 相互循环引用,即两个对象间的相互强引用.
- 自引用导致的循环引用.
__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;