objective-c中的特性

來源:互聯網
上載者:User

標籤:

一,retain, copy, assign區別 

概念: 
assign: 簡單賦值,不更改索引計數(reference counting)。 
copy: 建立一個索引計數為1的對象,然後釋放舊對象 
retain:釋放舊的對象,將舊對象的值賦予輸入對象,再提高輸入對象的索引計數為1 

例: 

1 NSString *pt = [[NSString alloc] initwithstring:@"abc"]; 

上面一段代碼會執行以下兩個動作 
1 在堆上分配一段記憶體用來儲存@"abc"  比如:記憶體位址為:0x1111 內容為 "abc" 
2 在棧上分配一段記憶體用來儲存pt  比如:地址為:0xaaaa 內容自然為0x1111   
下面分別看下assign retain copy 
assign的情況:NSString *newpt = [pt assign];     
此時newpt和pt完全相同 地址都是0xaaaa  內容為0x1111  即newpt只是pt的別名,對任何一個操作就等於對另一個操作。 因此retaincount不需要增加。 
retain的情況:NSString *newpt = [pt retain];     
此時newpt的地址不再為0xaaaa,可能為0xaabb 但是內容依然為0x1111。 因此newpt 和 pt 都可以管理"abc"所在的記憶體。因此 retaincount需要增加1   
copy的情況:NSString *newpt = [pt copy];   
此時會在堆上重新開闢一段記憶體存放@"abc" 比如0x1122 內容為@"abc 同時會在棧上為newpt分配空間 比如地址:0xaacc 內容為0x1122 因此retaincount增加1供newpt來管理0x1122這段記憶體 
  
理解: 
1,假設你用malloc 分配了一塊記憶體,並把它的地址賦值給了指標a,   後來你希望指標b也共用這段記憶體,於是你又把a賦值給  (assign)了 b。此時a和b指向同一塊記憶體,當a不需要使用這段記憶體時是不能直接釋放它,因為a並不知道 b也在使用這塊記憶體,如果 a釋放了,那麼b在使用這塊記憶體的時候程式會crash掉 
2,為瞭解決1中的問題,最簡單的一個方法就是使用引用計數,在上面的那個例子中,我們給那塊記憶體設一個引用計數,當記憶體被分配並且賦值給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各自有自己的記憶體。 

總結: 
使用assign: 對基礎資料類型 (NSInteger,CGFloat)和C資料類型(int, float, double, char, 等等) 
使用copy: 對NSString 
使用retain: 對其他NSObject和其子類 

二,ios5中新加入RAC的strong,week,unsafe_unretained 

說明: 
iOS5中新的關鍵字strong, weak, unsafe_unretained. 可以與以前的關鍵字對應學習strong與retain類似,weak與unsafe_unretained功能差不多(有點區別,等下會介紹,這兩個新關鍵字與assign類似)。在iOS5中用這些新的關鍵字,就可以不用手動管理記憶體了 
具體使用: 
strong關鍵字與retain關似,用了它,引用計數自動+1,用執行個體更能說明一切 

    1.    @property (nonatomic, strong) NSString *string1;   
    2.    @property (nonatomic, strong) NSString *string2;  

// 附註:nonatomic關鍵字: 

 noatomic是Objc使用的一種線程保護技術,基本上來講,是防止在寫未完成的時候被另外一個線程讀取,造成資料錯誤。而這種機制是耗費系統資源的,所以在iPhone這種小型裝置上,如果沒有使用多線程間的通訊編程,那麼nonatomic是一個非常好的選擇。 

有這樣兩個屬性, 

    1.    @synthesize string1;   
    2.    @synthesize string2;  

猜一下下面代碼將輸出什麼結果? 

    1.    self.string1 = @"String 1";   
    2.    self.string2 = self.string1;   
    3.    self.string1 = nil;  
    4.    NSLog(@"String 2 = %@", self.string2);  

結果是:String 2 = String 1 

由於string2是strong定義的屬性,所以引用計數+1,使得它們所指向的值都是@"String 1", 如果你對retain熟悉的話,這理解並不難。 

接著我們來看weak關鍵字: 
如果這樣聲明兩個屬性: 

    1.    @property (nonatomic, strong) NSString *string1;   
    2.    @property (nonatomic, weak) NSString *string2;  

並定義 

    1.    <pre name="code" class="cpp">@synthesize string1;   
    2.    @synthesize string2;  

 

再來猜一下,下面輸出是什嗎? 

    1.    self.string1 = @"String 1";   
    2.    self.string2 = self.string1;   
    3.    self.string1 = nil;  
    4.    NSLog(@"String 2 = %@", self.string2);  

結果是:String 2 = null 

分析一下,由於self.string1與self.string2指向同一地址,且string2沒有retain記憶體位址,而self.string1=nil釋放了記憶體,所以string1為nil。聲明為weak的指標,指標指向的地址一旦被釋放,這些指標都將被賦值為nil。這樣的好處能有效防止野指標。 

接著我們來看unsafe_unretained 
從名字可以看出,unretained且unsafe,由於是unretained所以與weak有點類似,但是它是unsafe的,什麼是unsafe的呢,下面看執行個體。 
如果這樣聲明兩個屬性: 
並定義 
    1.    @property (nonatomic, strong) NSString *string1;   
    2.    @property (nonatomic, unsafe_unretained) NSString *string2;  

再來猜一下,下面的代碼會有什麼結果? 
    1.    self.string1 = @"String 1";   
    2.    self.string2 = self.string1;   
    3.    self.string1 = nil;  
    4.    NSLog(@"String 2 = %@", self.string2);  

請注意,在此我並沒有叫你猜會有什麼輸出,因為根本不會有輸出,你的程式會crash掉。 

原因是什麼,其實就是野指標造成的,所以野指標是可怕的。為何會造成野指標呢?同於用unsafe_unretained聲明的指標,由於self.string1=nil已將記憶體釋放掉了,但是string2並不知道已被釋放了,所以是野指標。然後訪問野指標的記憶體就造成crash.  所以盡量少用unsafe_unretained關鍵字。 

strong,weak, unsafe_unretained往往都是用來聲明屬性的,如果想聲明臨時變數就得用__strong,  __weak, __unsafe_unretained,  __autoreleasing, 其用法與上面介紹的類似。 
還是看看執行個體吧。 

    1.    __strong NSString *yourString = @"Your String";   
    2.    __weak  NSString *myString = yourString;   
    3.    yourString = nil;   
    4.    __unsafe_unretained NSString *theirString = myString;  
    5.    //現在所有的指標都為nil  

再看一個: 

    1.    __strong NSString *yourString = @"Your String";   
    2.    __weak  NSString *myString = yourString;   
    3.    __unsafe_unretained NSString *theirString = myString;  
    4.    yourString = nil;   
    5.    //現在yourString與myString的指標都為nil,而theirString不為nil,但是是野指標。

objective-c中的特性

相關文章

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在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.