assign: 簡單賦值敦间,不更改索引計數(shù)
copy: 建立一個索引計數(shù)為1的對象瓶逃,然后釋放舊對象
retain:釋放舊的對象,將舊對象的值賦予輸入對象廓块,再提高輸入對象的索引計數(shù)為1
Copy其實是建立了一個相同的對象金闽,而retain不是:
比如一個NSString對象,地址為0×1111剿骨,內(nèi)容為@”STR”
Copy到另外一個NSString之 后,地址為0×2222埠褪,內(nèi)容相同浓利,新的對象retain為1, 舊有對象沒有變化
retain到另外一個NSString之 后钞速,地址相同(建立一個指針贷掖,指針拷貝),內(nèi)容當(dāng)然相同渴语,這個對象的retain值+1
也就是說苹威,retain是指針拷貝,copy是內(nèi)容拷貝驾凶。在拷貝之前牙甫,都會釋放舊的對象。
- 使用assign: 對基礎(chǔ)數(shù)據(jù)類型 (NSInteger)和C數(shù)據(jù)類型(int, float, double, char,等)
- 使用copy: 對NSString
- 使用retain: 對其他NSObject和其子類
1.readonly表示這個屬性是只讀的调违,就是只生成getter方法窟哺,不會生成setter方法.
2.readwrite,設(shè)置可供訪問級別
3.retain技肩,是說明該屬性在賦值的時候且轨,先release之前的值,然后再賦新值給屬性虚婿,引用再加1旋奢。
4.nonatomic,非原子性訪問然痊,不加同步至朗,多線程并發(fā)訪問會提高性能。注意剧浸,如果不加此屬性爽丹,則默認(rèn)是兩個訪問方法都為原子型事務(wù)訪問筑煮。
retain和copy還有assign的區(qū)別
假設(shè)你用malloc分配了一塊內(nèi)存,并且把它的地址賦值給了指針a粤蝎,后來你希望指針b也共享這塊內(nèi)存真仲,于是你又把a賦值給(assign)了b。此時a和b指向同一塊內(nèi)存初澎,請問當(dāng)a不再需要這塊內(nèi)存秸应,能否直接釋放它?答案是否定的碑宴,因為a并不知道b是否還在使用這塊內(nèi)存软啼,如果a釋放了,那么b在使用這塊內(nèi)存的時候會引起程序crash掉延柠。
了解到1中assign的問題祸挪,那么如何解決?最簡單的一個方法就是使用引用計數(shù)(reference counting)贞间,還是上面的那個例子贿条,我們給那塊內(nèi)存設(shè)一個引用計數(shù),當(dāng)內(nèi)存被分配并且賦值給a時增热,引用計數(shù)是1整以。當(dāng)把a賦值給b時引用計數(shù)增加到2。這時如果a不再使用這塊內(nèi)存峻仇,它只需要把引用計數(shù)減1公黑,表明自己不再擁有這塊內(nèi)存。b不再使用這塊內(nèi)存時也把引用計數(shù)減1摄咆。當(dāng)引用計數(shù)變?yōu)?的時候凡蚜,代表該內(nèi)存不再被任何指針?biāo)茫到y(tǒng)可以把它直接釋放掉吭从。
上面兩點其實就是assign和retain的區(qū)別番刊,assign就是直接賦值,從而可能引起1中的問題影锈,當(dāng)數(shù)據(jù)為int, float等原生類型時芹务,可以使用assign。retain就如2中所述鸭廷,使用了引用計數(shù)枣抱,retain引起引用計數(shù)加1, release引起引用計數(shù)減1,當(dāng)引用計數(shù)為0時辆床,dealloc函數(shù)被調(diào)用佳晶,內(nèi)存被回收。
copy是在你不希望a和b共享一塊內(nèi)存時會使用到讼载。a和b各自有自己的內(nèi)存轿秧。
atomic和nonatomic用來決定編譯器生成的getter和setter是否為原子操作中跌。在多線程環(huán)境下,原子操作是必要的菇篡,否則有可能引起錯誤的結(jié)果漩符。加了atomic,setter函數(shù)會變成下面這樣:
if (property != newValue) {
[property release];
property = [newValue retain];
}
關(guān)于retain,copy,assign的區(qū)別問題其實困擾我很久了驱还,因為在程序中不太常用到copy嗜暴,assign,所以三者的具體差別一直不太明白议蟆。
按照我的理解闷沥,assign和retain的區(qū)別,就是引入了一個計數(shù)器retaincount咐容,就可以對一個內(nèi)存的釋放方便很多舆逃。copy,就是把原來的內(nèi)存復(fù)制一遍戳粒,使各自都擁有一個內(nèi)存路狮,這樣釋放的時候也不會出錯。
assign: 簡單賦值享郊,不更改索引計數(shù)(Reference Counting)。
copy: 建立一個索引計數(shù)為1的對象孝鹊,然后釋放舊對象
retain:釋放舊的對象炊琉,將舊對象的值賦予輸入對象,再提高輸入對象的索引計數(shù)為1
使用assign: 對基礎(chǔ)數(shù)據(jù)類型 (NSInteger又活,CGFloat)和C數(shù)據(jù)類型(int, float, double, char, 等等)
使用copy: 對NSString
使用retain: 對其他NSObject和其子類
nonatomic苔咪,非原子性訪問,不加同步柳骄,多線程并發(fā)訪問會提高性能团赏。注意,如果不加此屬性耐薯,則默認(rèn)是兩個訪問方法都為原子型事務(wù)訪問
@property(nonatomic, retain) UITextField *userName編譯時自動生成的代碼
(UITextField *) userName {
return userName;
}(void) setUserName:(UITextField *)userName_ {
[userName release];
userName = [userName_ retain];
}
@property(retain) UITextField *userName自動生成的代碼
(UITextField *) userName {
UITextField *retval = nil;
@synchronized(self) {
retval = [[userName retain] autorelease];
}
return retval;
}(void) setUserName:(UITextField *)userName_ {
@synchronized(self) {
[userName release];
userName = [userName_ retain];
}
}