
atomic與nonatomic用來決定編譯器生成的getter和setter是否為原子操作。 atomic:設置成員變量的@PRoperty屬性時,默認為atomic,提供多線程安全屬性。在多線程環境下,原子操作是必要的,否則會引起莫名其妙的BUG。加了atomic,setter函數相當于如下:
{lock}if (property != newValue) { [property release]; property = [newValue retain];}{unlock}nonatomic:禁止多線程,變量保護,提高性能。 atomic是Objc使用的一種線程保護技術,基本上來講,是防止在寫未完成的時候被另外一個線程讀取,造成數據異常。而這種機制是耗費資源的,所以在iphone這種小型設備上,如果沒有使用多線程間的通訊編程,nonatomic是一個優化的選擇。
對基礎數據類型 (NSInteger,CGFloat)和C數據類型(int, float, double, char)等。簡單賦值,不更改索引計數。 此標記說明設置器直接進行賦值,這也是默認值。在使用垃圾收集的應用程序中,如果你要一個屬性使用assign,且這個類符合NSCopying協議,你就要明確指出這個標記,而不是簡單地使用默認值,否則的話,你將得到一個編譯警告。這再次向編譯器說明你確實需要賦值,即使它是可拷貝的。
NSObject和其子類對參數進行release舊值,再retain新值 指定retain會在賦值時喚醒傳入值的retain消息。此屬性只能用于Objective-C對象類型,而不能用于Core Foundation對象。(原因很明顯,retain會增加對象的引用計數,而基本數據類型或者Core Foundation對象都沒有引用計數)。 注意: 把對象添加到數組中時,引用計數將增加對象的引用次數+1。
對NSString 它指出,在賦值時使用傳入值的一份拷貝。拷貝工作由copy方法執行,此屬性只對那些實行了NSCopying協議的對象類型有效。
copy與retain: Copy其實是建立了一個相同的對象,而retain不是: 1、比如一個NSString 對象,地址為0×1111 ,內容為@”STR”,Copy 到另外一個NSString 之后,地址為0×2222 ,內容相同。 2、新的對象retain為1 ,舊有對象沒有變化retain 到另外一個NSString 之后,地址相同(建立一個指針,指針拷貝),內容當然相同,這個對象的retain值+1。 Copy屬性表示兩個對象內容相同,新的對象retain為1 ,與舊有對象的引用計數無關,舊有對象沒有變化。copy減少對象對上下文的依賴。 Retain屬性表示兩個對象地址相同(建立一個指針,指針拷貝),內容當然相同,這個對象的retain值+1也就是說,retain 是指針拷貝,copy 是內容拷貝。當然在iOS中并不是所有的對象都支持copy,mutableCopy,遵守NSCopying 協議的類可以發送copy消息,遵守NSMutableCopying 協議的類才可以發送mutableCopy消息。假如發送了一個沒有遵守上訴兩協議而發送 copy或者 mutableCopy,那么就會發生異常。但是默認的ios類并沒有遵守這兩個協議。如果想自定義一下copy 那么就必須遵守NSCopying,并且實現 copyWithZone: 方法,如果想自定義一下mutableCopy 那么就必須遵守NSMutableCopying,并且實現 mutableCopyWithZone: 方法。 總結:retain 是指針拷貝,copy 是內容拷貝。
assign與retain: 1. 接觸過C,那么假設你用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的時候,代表該內存不再被任何指針所引用,系統可以把它直接釋放掉。 總結:上面兩點其實就是assign和retain的區別,assign就是直接賦值,從而可能引起1中的問題,當數據為int, float等原生類型時,可以使用assign。retain就如2中所述,使用了引用計數,retain引起引用計數加1, release引起引用計數減1,當引用計數為0時,dealloc函數被調用,內存被回收。
新聞熱點
疑難解答