在iOS開發(fā)過程中代态,屬性的定義往往與retain, assign, copy有關(guān),這里先簡單介紹下這幾個的區(qū)別
NSString?*pt?=?[[NSString?alloc]?initWithString:@"abc"];
上面一段代碼會執(zhí)行以下兩個動作
1?在堆上分配一段內(nèi)存用來存儲@"abc"??比如:內(nèi)存地址為:0X1111?內(nèi)容為?"abc"
2?在棧上分配一段內(nèi)存用來存儲pt??比如:地址為:0Xaaaa?內(nèi)容自然為0X1111
下面分別看下assign?retain?copy
assign的情況:NSString?*newPt?=?[pt?assing];
此時newPt和pt完全相同?地址都是0Xaaaa??內(nèi)容為0X1111??即newPt只是pt的別名寺惫,對任何一個操作就等于對另一個操作。?因此retainCount不需要增加蹦疑。
retain的情況:NSString?*newPt?=?[pt?retain];
此時newPt的地址不再為0Xaaaa西雀,可能為0Xaabb?但是內(nèi)容依然為0X1111。?因此newPt?和?pt?都可以管理"abc"所在的內(nèi)存歉摧。因此?retainCount需要增加1
copy的情況:NSString?*newPt?=?[pt?copy];
此時會在堆上重新開辟一段內(nèi)存存放@"abc"?比如0X1122?內(nèi)容為@"abc?同時會在棧上為newPt分配空間?比如地址:0Xaacc?內(nèi)容為0X1122?因此retainCount增加1供newPt來管理0X1122這段內(nèi)存
現(xiàn)在我們看看iOS5中新的關(guān)鍵字strong, weak, unsafe_unretained. 可以與以前的關(guān)鍵字對應(yīng)學(xué)習(xí)strong與retain類似蒋搜,weak與unsafe_unretained功能差不多(有點區(qū)別篡撵,等下會介紹,這兩個新 關(guān)鍵字與assign類似)豆挽。在iOS5中用這些新的關(guān)鍵字育谬,就可以不用手動管理內(nèi)存了,從java等其它語言轉(zhuǎn)過來的程序員非常受用帮哈。
strong關(guān)鍵字與retain關(guān)似膛檀,用了它,引用計數(shù)自動+1娘侍,用實例更能說明一切
@property?(nonatomic,?strong)?NSString?*string1;
@property?(nonatomic,?strong)?NSString?*string2;
有這樣兩個屬性
@synthesize?string1;
@synthesize?string2;
猜一下下面代碼將輸出什么結(jié)果咖刃?
self.string1?=?@"String?1";
[self.string2?=?self.string1;
[self.string1?=?nil;
[NSLog(@"String?2?=?%@",?self.string2);
結(jié)果是:String 2 = String 1
由于string2是strong定義的屬性,所以引用計數(shù)+1憾筏,使得它們所指向的值都是@"String 1", 如果你對retain熟悉的話嚎杨,這理解并不難。
接著我們來看weak關(guān)鍵字:
如果這樣聲明兩個屬性:
@property?(nonatomic,?strong)?NSString?*string1;
@property?(nonatomic,?weak)?NSString?*string2;
并定義
@synthesize?string1;
@synthesize?string2;
再來猜一下氧腰,下面輸出是什么枫浙?
self.string1?=?[[NSString?alloc]?initWithUTF8String:"string?1"];
elf.string2?=?self.string1;
self.string1?=?nil;
NSLog(@"String?2?=?%@",?self.string2);
結(jié)果是:String 2 = null
分析一下,由于 self.string1與self.string2指向同一地址古拴,且string2沒有retain內(nèi)存地址箩帚,而self.string1=nil釋放 了內(nèi)存,所以string1為nil黄痪。聲明為weak的指針紧帕,指針指向的地址一旦被釋放,這些指針都將被賦值為nil桅打。這樣的好處能有效的防止野指針是嗜。在 c/c++開發(fā)過程中,為何大牛都說指針的空間釋放了后挺尾,都要將指針賦為NULL. 在這兒用weak關(guān)鍵字幫我們做了這一步叠纷。
接著我們來看unsafe_unretained
從名字可以看出,unretained且unsafe潦嘶,由于是unretained所以與weak有點類似,但是它是unsafe的崇众,什么是unsafe的呢掂僵,下面看實例。
如果這樣聲明兩個屬性:
并定義
@property?(nonatomic,?strong)?NSString?*string1;
@property?(nonatomic,?unsafe_unretained)?NSString?*string2;
再來猜一下顷歌,下面的代碼會有什么結(jié)果锰蓬?
self.string1?=?[[NSString?alloc]?initWithUTF8String:"string?1"];
self.string2?=?self.string1;
self.string1?=?nil;
NSLog(@"String?2?=?%@",?self.string2);
請注意,在此我并沒有叫你猜會有什么輸出眯漩,因為根本不會有輸出芹扭,你的程序會crash掉麻顶。 原因是什么,其實 就是野指針造成的舱卡,所以野指針是可怕的辅肾。為何會造成野指針呢?同于用unsafe_unretained聲明的指針轮锥,由于 self.string1=nil已將內(nèi)存釋放掉了矫钓,但是string2并不知道已被釋放了,所以是野指針舍杜。然后訪問野指針的內(nèi)存就造成crash. ?所以盡量少用unsafe_unretained關(guān)鍵字新娜。