iOS中assign、copy 、retain等關鍵字的含義

來源:互聯網
上載者:User

原文出處:http://www.2cto.com/kf/201205/133943.html

assign: 簡單賦值,不更改索引計數

copy: 建立一個索引計數為1的對象,然後釋放舊對象
retain:釋放舊的對象,將舊對象的值賦予輸入對象,再提高輸入對象的索引計數為1

Copy其實是建立了一個相同的對象,而retain不是:

比如一個NSString對象,地址為0×1111,內容為@”STR”
Copy到另外一個NSString之 後,地址為0×2222,內容相同,新的對象retain為1, 舊有對象沒有變化

retain到另外一個NSString之 後,地址相同(建立一個指標,指標拷貝),內容當然相同,這個對象的retain值+1

也就是說,retain是指標拷貝,copy是內容拷貝。在拷貝之前,都會釋放舊的對象。

* 使用assign: 對基礎資料類型 (NSInteger)和C資料類型(int, float, double, char,等)

* 使用copy: 對NSString
* 使用retain: 對其他NSObject和其子類

1.readonly,表示這個屬性是唯讀,就是只產生getter方法,不會產生setter方法.

2.readwrite,設定可供存取層級
3.retain,是說明該屬性在賦值的時候,先release之前的值,然後再賦新值給屬性,引用再加1。
4.nonatomic,非原子性訪問,不加同步,多線程並發訪問會提高效能。注意,如果不加此屬性,則預設是兩個存取方法都為原子型事務訪問。

retain和copy還有assign的區別

1. 假設你用malloc分配了一塊記憶體,並且把它的地址賦值給了指標a,後來你希望指標b也共用這塊記憶體,於是你又把a賦值給(assign)了b。此時a和b指向同一塊記憶體,請問當a不再需要這塊記憶體,能否直接釋放它?答案是否定的,因為a並不知道b是否還在使用這塊記憶體,如果a釋放了,那麼b在使用這塊記憶體的時候會引起程式crash掉。

2. 瞭解到1中assign的問題,那麼如何解決?最簡單的一個方法就是使用引用計數(reference counting),還是上面的那個例子,我們給那塊記憶體設一個引用計數,當記憶體被分配並且賦值給a時,引用計數是1。當把a賦值給b時引用計數增加到2。這時如果a不再使用這塊記憶體,它只需要把引用計數減1,表明自己不再擁有這塊記憶體。b不再使用這塊記憶體時也把引用計數減1。當引用計數變為0的時候,代表該記憶體不再被任何指標所引用,系統可以把它直接釋放掉。

3. 上面兩點其實就是assign和retain的區別,assign就是直接賦值,從而可能引起1中的問題,當資料為int, float等原生類型時,可以使用assign。retain就如2中所述,使用了引用計數,retain引起引用計數加1, release引起引用計數減1,當引用計數為0時,dealloc函數被調用,記憶體被回收。

4. copy是在你不希望a和b共用一塊記憶體時會使用到。a和b各自有自己的記憶體。
5. atomic和nonatomic用來決定編譯器產生的getter和setter是否為原子操作。在多線程環境下,原子操作是必要的,否則有可能引起錯誤的結果。加了atomic,setter函數會變成下面這樣:
if (property != newValue) {
[property release];
property = [newValue retain];
}

關於retain,copy,assign的區別問題其實困擾我很久了,因為在程式中不太常用到copy,assign,所以三者的具體差別一直不太明白。

按照我的理解,assign和retain的區別,就是引入了一個計數器retaincount,就可以對一個記憶體的釋放方便很多。copy,就是把原來的記憶體複製一遍,使各自都擁有一個記憶體,這樣釋放的時候也不會出錯。
assign: 簡單賦值,不更改索引計數(Reference Counting)。
copy: 建立一個索引計數為1的對象,然後釋放舊對象
retain:釋放舊的對象,將舊對象的值賦予輸入對象,再提高輸入對象的索引計數為1
使用assign: 對基礎資料類型 (NSInteger,CGFloat)和C資料類型(int, float, double, char, 等等)
使用copy: 對NSString
使用retain: 對其他NSObject和其子類
nonatomic,非原子性訪問,不加同步,多線程並發訪問會提高效能。注意,如果不加此屬性,則預設是兩個存取方法都為原子型事務訪問

@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];
}

}

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.