被无数人写过的assign,retain,strong,weak,unsafe_unretained,还有copy

虽然的确是最基本&被无数人写过的问题,但是今天还是想弄得更清楚一些,所以看了看官方文档,写了这篇博客。

assign,retain,strong,weak,unsafe_unretained,还有copy,这些都是一个property在声明中可以指定的属性,且都与内存管理有关。下面会从Non-ARC和ARC两种情况讨论一下这些属性的意义。


Non-ARC



官方文档的描述上看,Non-ARC的内存管理模式下,编译器会为带有不同属性的property自动生成对应的accessor方法。并且苹果十分建议在可能的情况下通过accessor方法来操纵property,而不是操纵它对应的实例变量。

如果需要对某些property自定义accessor方法,则需要程序员注意这个property的属性。个人认为,写在property旁边的属性,并不是真正控制着这个property的行为,它只是对编译器自动生成的accessor方法提供了指导,当然也为自定义accessor方法的程序员和他的客户程序员提供了指导。

  • assign

在Non-ARC内存管理模式下,assign是一个property的默认属性,无论这个property代表一个简单数据类型,还是一个指向对象的指针。也就是说:

@property (nonatomic) NSNumber *count;

等价于:

@property (nonatomic, assign) NSNumber *count;

assign主要应用于代表简单数据类型的property,比如int,float等。
如果这个用assign属性修饰的property代表一个指向对象的指针,那么当这个指针指向某个对象时,这个对象的引用计数不应该被改变。也就是说,用assign属性修饰的property,不应该持有一个对象。
因为这个property不持有对象,所以它所指向的对象很可能已经在别处被释放了。这时它就有可能成为一枚悬垂指针,访问它指向的内存地址时,可能会发生意想不到的状况。

  • retain

retain不能修饰用来代表简单数据类型的property,否则编译器会报错:

@property (nonatomic, retain) int num;//编译器报错:Property with 'retain (or strong)' attribute must be of object type

如果一个property被retain修饰,这代表着这个property应该持有它所指向的对象。
官方文档中展示了一个被retain修饰的property:

@property (nonatomic, retain) NSNumber *count;

编译器可能为它实现的accessor方法:

- (NSNumber *)count {
    return _count;
}
- (void)setCount:(NSNumber *)newCount {
    [newCount retain];
    [_count release];
    // Make the new assignment.
    _count = newCount;
}

注意,考虑到newCount和_count可能指向同一个对象,所以在setter方法中,必须首先调用retain,以防这个对象被释放。

  • copy

copy也不能修饰用来代表简单数据类型的property,否则编译器会报错:

@property (nonatomic, copy) int num;//编译器报错:Property with 'copy' attribute must be of object type

如果一个property被copy修饰,那么赋值到这个property的对象,应该是原有对象的一份拷贝。
只有实现了NSCopying协议,并且实现了其中的copyWithZone:方法的对象才能被拷贝。
但是并不是所有的拷贝都产生了新的对象,有些类在实现copyWithZone:方法时,有着它们自己的考虑。比如NSString

@property (nonatomic, copy) NSString *myString;
NSString *string = [[NSString alloc] initWithString:@"Hello"];
self.myString = string;
NSLog(@"%d", string == _myString);//输出1

在这里,property的指针和原先的指针指向的是同一个地址。

  • unsafe_unretained

个人认为unsafe_unretained与assign是等价的。

  • strong

个人认为strong与retain是等价的。
官方文档中有这样的示例代码:

// The following declaration is a synonym for: @property(retain) MyClass *myObject;
@property(strong) MyClass *myObject;

表示了strong和retain是同义词。

  • weak

Non-ARC内存管理模式下无法使用weak来修饰一个property,编译器会报错。


ARC



ARC有效时,对象类型的变量将有所有权修饰符来修饰。一共有以下四种所有权修饰符:

__strong 修饰符
__weak 修饰符
__unsafe_unretained 修饰符
__autoreleasing 修饰符

四种修饰符的具体意思,就不在这里解释了(´・_・`)

编译器在为一个property合成实例变量时,也会使用所有权修饰符来修饰这个实例变量。根据property属性的不同,用来修饰实例变量的所有权修饰符也不尽相同。

  • strong

在ARC内存管理模式下,strong是一个代表对象类型的property的默认属性,并且它不能修饰用来代表简单数据类型的property。编译器在合成实例变量时,将使用__strong修饰符。
如果另外自定义了用其他修饰符修饰的实例变量,编译器会报错。可以用这个方法来验证property的各个属性对应的实例变量的所有权修饰符。

@interface ViewController ()
{
    __weak NSObject *_obj;//编译器报错:Existing instance variable '_obj' for strong property 'obj' may not be weak
}

@property (nonatomic, strong) NSObject *obj;

@end
  • weak

weak也不能修饰用来代表简单数据类型的property。
编译器将为weak修饰的property生成带__weak所有权修饰符的实例变量。

  • copy

copy也不能修饰用来代表简单数据类型的property。
编译器将为copy修饰的property生成带__strong所有权修饰符的实例变量。
编译器自动合成的setter方法会调用对象的copyWithZone:方法。虽然第三方程序员可以自定义setter方法,但是为了程序的可读性,也应该在其中执行拷贝的逻辑。

  • retain

和Non-ARC的理由一样,个人认为retain和strong是等价的。

  • unsafe_unretained

编译器将为unsafe_unretained修饰的property生成带__unsafe_unretained所有权修饰符的实例变量。
与weak和strong不同的是,unsafe_unretained也可以修饰代表简单数据类型的property。

  • assign

个人认为assign和unsafe_unretained等价。
assign在ARC内存管理模式下,仍然是代表简单数据类型的property的默认属性。


参考:

Transitioning to ARC Release Notes
Practical Memory Management
Objective-C Automatic Reference Counting (ARC)
Object copying
Objective-C高级编程 iOS与OS X多线程和内存管理

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 141,022评论 1 296
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 60,531评论 1 254
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 92,811评论 0 210
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 40,662评论 0 171
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 48,341评论 1 250
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 38,549评论 1 169
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 30,248评论 2 266
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 29,023评论 0 161
  • 想象着我的养父在大火中拼命挣扎,窒息,最后皮肤化为焦炭。我心中就已经是抑制不住地欢快,这就叫做以其人之道,还治其人...
    爱写小说的胖达阅读 28,764评论 6 225
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 32,375评论 0 211
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 29,146评论 2 212
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 30,454评论 1 222
  • 白月光回国,霸总把我这个替身辞退。还一脸阴沉的警告我。[不要出现在思思面前, 不然我有一百种方法让你生不如死。]我...
    爱写小说的胖达阅读 24,146评论 0 31
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 26,992评论 2 210
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 31,364评论 3 201
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 25,575评论 0 8
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 25,918评论 0 163
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 33,329评论 2 227
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 33,447评论 2 229

推荐阅读更多精彩内容