assign:
一般用于基本數(shù)據(jù)類型镣陕、枚舉秸苗、結(jié)構(gòu)體等非OC對象類型伪煤。
copy:
一般用于 NSString 類型损离、block 上哼审。
為什么 NSString 使用 copy 而不使用 strong碘耳?
區(qū)別不大显设,個人覺得可以防止 NSMutableString 被無意中修改。
修改方法見下面介紹的的關(guān)于 strong 和 copy 的一些區(qū)別
strong:
strong 與 weak 是因為 ARC 出現(xiàn)而新引入的對象變量屬性藏畅。
strong 和 retain 同義, weak 和 assign 同義敷硅。
一般用于 OC 對象類型(NSArray、NSDate愉阎、NSNumber绞蹦、模型類)和 UI 控件
關(guān)于 strong 和 copy 的一些區(qū)別
下面舉一個例子
@property (nonatomic, copy) NSString *strCopy1;
@property (nonatomic, copy) NSString *strCopy2;
@property (nonatomic, copy) NSString *strCopy3;
@property (nonatomic, strong) NSString *strStrong;
NSMutableString *str = [NSMutableString stringWithFormat:@"1"];
_strStrong = str;
_strCopy1 = str;
self.strCopy2 = str;
_strCopy3 = [str copy];
NSLog(@"_strStrong:%@ 地址:%p", _strStrong, &_strStrong);
NSLog(@"_strCopy1:%@ 地址:%p", _strCopy1, &_strCopy1);
NSLog(@"_strCopy2:%@ 地址:%p", _strCopy2, &_strCopy2);
NSLog(@"_strCopy3:%@ 地址:%p", _strCopy3, &_strCopy3);
[str appendString:@"2"];
NSLog(@"_strStrong:%@ 地址:%p", _strStrong, &_strStrong);
NSLog(@"_strCopy1:%@ 地址:%p", _strCopy1, &_strCopy1);
NSLog(@"_strCopy2:%@ 地址:%p", _strCopy2, &_strCopy2);
NSLog(@"_strCopy3:%@ 地址:%p", _strCopy3, &_strCopy3);
輸出結(jié)果:
_strStrong:1 地址:0x7ff31cc0c3d0
_strCopy1:1 地址:0x7ff31cc0c3b8
_strCopy2:1 地址:0x7ff31cc0c3c0
_strCopy3:1 地址:0x7ff31cc0c3c8
_strStrong:12 地址:0x7ff31cc0c3d0
_strCopy1:12 地址:0x7ff31cc0c3b8
_strCopy2:1 地址:0x7ff31cc0c3c0
_strCopy3:1 地址:0x7ff31cc0c3c8
結(jié)論:
1.用等號直接賦值,會把內(nèi)存地址一起復制過去榜旦。所以 str 改變時 _strStrong 也會跟著改變幽七。
2.strCopy1 雖然聲明為 copy 屬性,但是沒有用 self 的點語法溅呢,所以沒有調(diào)用到自身的 set 方法澡屡≡持浚可參考 strCopy2 的輸出。
3.strCopy3 是手動使用 copy 方法驶鹉,所以也可以得到 1 這個結(jié)果绩蜻。
weak:
一般用于 delegate,weak 比 assign 多了一個功能室埋,當對象消失后自動把指針變成 nil办绝。
在防止如 block 的循環(huán)引用時,使用 __weak
關(guān)鍵字做如下定義:
__weak typeof(self) weakSelf = self;
也可以定義一個宏:
#define WeakSelf __weak typeof(self) weakSelf = self;
atomic:
1.默認屬性
2.安全姚淆。會保證 CPU 不會出現(xiàn)某一線程執(zhí)行完 setter 全部語句之前孕蝉,另一個線程開始執(zhí)行 setter 情況
3.速度不快,因為要保證操作整體完成
nonatomic:
1.非默認屬性腌逢,但是最常用
2.線程不安全降淮,如有兩個線程訪問同一個屬性,會出現(xiàn)無法預料的結(jié)果
3.速度更快
atomic 和 nonatomic 自動生成的 set 方法詳解:
// @property(nonatomic, retain) UITextField *userName;
// 系統(tǒng)生成的代碼如下:
- (UITextField *)userName {
return userName;
}
- (void)setUserName:(UITextField *)userName_ {
[userName_ retain];
[userName release];
userName = userName_;
}
而 atomic 版本的要復雜一些:
// @property(retain) UITextField *userName;
// 系統(tǒng)生成的代碼如下:
- (UITextField *)userName {
UITextField *retval = nil;
@synchronized(self) {
retval = [[userName retain] autorelease];
}
return retval;
}
- (void)setUserName:(UITextField *)userName_ {
@synchronized(self) {
[userName release];
userName = [userName_ retain];
}
}
簡單來說搏讶,就是 atomic 會加一個鎖來保障線程安全佳鳖,并且引用計數(shù)會 +1宇葱,來向調(diào)用者保證這個對象會一直存在祝旷。假如不這樣做溢十,如有另一個線程調(diào) setter摹芙,可能會出現(xiàn)線程競態(tài)稍味,導致引用計數(shù)降到 0舵变,原來那個對象就釋放掉了苟鸯。