NSString *pt = [[NSString alloc] initWithString:@"abc"];
上面一段代码会执行以下两个动作
1 在堆上分配一段内存用来存储@"abc" 比如:内存地址为:0x1111 内容为 "abc" 2 在栈上分配一段内存用来存储pt 比如:地址为:0xAAAA内容自然为0x1111
下面分别看下assign retain copy
assign的情况:NSString *newPt = [pt assing];
此时newPt和pt完全相同地址都是0xAAAA内容为0x1111即newPt只是pt的别名,对任何一个操作就等于对另一个操作。因此retainCount不需要增加。
retain的情况:NSString *newPt = [pt retain];
此时newPt的地址不再为0xAAAA,可能为0xAABB但是内容依然为0x1111。因此newPt 和 pt 都可以管理"abc"所在的内存。因此 retainCount需要增加1
copy的情况:NSString *newPt = [pt copy];
此时会在堆上重新开辟一段内存存放@"abc" 比如0x1122 内容为@"abc 同时会在栈上为newPt分配空间比如地址:0xAACC内容为0x1122 因此retainCount增加1供newPt来管理0x1122这段内存
strong和weak
自从有了ARC,就可以使用weak或strong来说明属性是弱引用还是强引用;
属性:
UI控件用weak,NSString用copy,其他对象用strong
如果报错出现copyWithZone 检查所有copy修饰的属性