iOS 關鍵字 copy,assign,strong,retain,weak,readonly,readwrite,nonatomic,atomic,unsafe_unretained的使用與區別

//聯系人:石虎QQ: 1224614774昵稱:嗡嘛呢叭咪哄

OC?關鍵字:copy,assign,strong,retain,weak,readonly,readwrite,nonatomic,atomic,unsafe_unretained的使用與區別

一、關鍵字的概念:

ARC-自動醫用技術

ARC不是垃圾回收,而是編譯器自動插入代碼來減少程序員的代碼輸入和失誤。

同時比垃圾和效率要高,因為其不影響運行時間,相當于自己管理內存。

總是通過屬性來管理實例變量(init/dealloc除外),在dealloc中釋放所有屬性。

dealloc中會自動加入釋放實例變量的代碼,因此不必要手段增加釋放實例變量的代碼。不需要手動調用[super? dealloc]

不要調用retain,release,autorelease,編譯器會自動插入相關代碼。

注意命名方式,不要以copyXXX方式命名不想進行retain的方法,編譯器會根據方法名自動retain。

C語言結構體中不要有對象指針

id和void*只能通過橋接轉換來進行轉換

不要使用NSAutoreleasePool,而是使用@autoreleasepool{}代碼塊。

轉換ARC代碼:Edit->Refactor->Convert? to Objective-C ARC

retain cycle

循環保留

delegate和block是產生retain? cycle的主要原因

dealloc

移除觀察者observers

注銷通知notification

設置非weak的delegate為nil??? 取消timer

二、關鍵字含義:

copy

復制內容(深復制),如果調用copy的是數組,則為指針復制(淺復制),僅僅復制子元素的指針。

@property? (nonatomic,copy)NSString? *title;

@property (nonatomic, copy) NSMutableArray *myArray;//not recommended

@property (nonatomic, copy) SomeBlockType someBlock;

assign

對基礎數據類型(NSInteger,CGFloat)和C數據類型(int,? float, double, char等)

@property? (nonatomic, assign) int n;

@property (nonatomic, assign) BOOL isOK;

@property (nonatomic,? assign)? CGFloat scalarFloat;

@property (nonatomic,? assign)? CGPoint scalarStruct;

strong

相當于retain。

Strong在ARC環境為默認屬性類型。

@property? (nonatomic,readwrite,strong)NSString *title;

@property (strong, nonatomic) UIViewController *viewController;

@property (nonatomic,? strong) id? childObject;

retain

NSObject及其子類。

Release舊值,retain新值。

Retain是指針復制(淺復制),引用計數加1,而不會導致內容被復制。

@property? (nonatomic, retain)UIColor *myColor;

weak

取代之前的assign,對象銷毀之后會自動置為nil,防止野指針。

Assign不能自動置為nil,需要手動置為nil。

Delegate基本總是使用weak,以防止循環引用。特殊情況是,希望在dealloc中調用delegate的某些方法進行釋放,此時如果使用weak將引起異常,因為此時已經是nil了,那么采用assign更為合適。

@property? (weak, nonatomic) IBOutlet UIButton *myButton;//處于最頂層的IBOutlet應該為strong

@property (nonatomic,? weak) id? parentObject;

@property(nonatomic,readwrite,weak) id? delegate;

@property (nonatomic,? weak) NSObject? *delegate;

readonly

此標記說明屬性是只讀的,默認的標記是讀寫,如果你指定了只讀,在@implementation中只需要一個讀取器。或者如果你使用@synthesize關鍵字,也是有讀取器方法被解析。而且如果你試圖使用點操作符為屬性賦值,你將得到一個編譯錯誤。

readwrite

此標記說明屬性會被當成讀寫的,這也是默認屬性。設置器和讀取器都需要在@implementation中實現。如果使用@synthesize關鍵字,讀取器和設置器都會被解析。

unsafe_unretained

unretained且unsafe,由于是unretained所以與weak有點類似,但是它是unsafe的.

@property(nonatomic,unsafe_unretained)Book *book1;

unsafe_unretainedid safeSelf = self;

三.關鍵字使用區別:

1:copy與retain:

1、copy其實是建立了一個相同的對象,而retain不是;

2、copy是內容拷貝,retain是指針拷貝;

3、copy是內容的拷貝 ,對于像NSString,的確是這樣,但是如果copy的是一個NSArray呢?這時只是copy了指向array中相對應元素的指針.這便是所謂的"淺復制".

4、copy的情況:NSString *newPt = [pt copy];

此時會在堆上重新開辟一段內存存放@"abc" 比如0X1122 內容為@"abc 同時會在棧上為newPt分配空間 比如地址:0Xaacc 內容為0X1122 因此retainCount增加1供newPt來管理0X1122這段內存;

二:assign與retain:

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

2、assign的情況:NSString *newPt = [pt assing];

此時newPt和pt完全相同 地址都是0Xaaaa 內容為0X1111 即newPt只是pt的別名,對任何一個操作就等于對另一個操作, 因此retainCount不需要增加;

3、assign就是直接賦值;

4、retain使用了引用計數,retain引起引用計數加1, release引起引用計數減1,當引用計數為0時,dealloc函數被調用,內存被回收;

5、retain的情況:NSString *newPt = [pt retain];

此時newPt的地址不再為0Xaaaa,可能為0Xaabb 但是內容依然為0X1111。 因此newPt 和 pt 都可以管理"abc"所在的內存,因此 retainCount需要增加1;

三:readonly:

readonly:只產生簡單的getter,沒有setter。

四:readwrite:

readwrite:同時產生setter\getter方法

五nonatomic,atomic:

:1、非原子性訪問,對屬性賦值的時候不加鎖,多線程并發訪問會提高性能。如果不加此屬性,則默認是兩個訪問方法都為原子型事務訪問;

weak and strong property (強引用和弱引用的區別):

:2,置成員變量的@property屬性時,默認為atomic,提供多線程安全。在多線程環境下,原子操作是必要的,否則有可能引起錯誤的結果

atomic的意思就是setter/getter這個函數,是一個原語操作。如果有多個線程同時調用setter的話,不會出現某一個線程執行完setter全部語句之前,另一個線程開始執行setter情況,相當于函數頭尾加了鎖一樣,可以保證數據的完整性。nonatomic不保證setter/getter的原語行,所以你可能會取到不完整的東西。因此,在多線程的環境下原子操作是非常必要的,否則有可能會引起錯誤的結果。

比如setter函數里面改變兩個成員變量,如果你用nonatomic的話,getter可能會取到只更改了其中一個變量時候的狀態,這樣取到的東西會有問題,就是不完整的。當然如果不需要多線程支持的話,用nonatomic就夠了,因為不涉及到線程鎖的操作,所以它執行率相對快些。

六:weak 和 strong

1、weak 和 strong 屬性只有在你打開ARC時才會被要求使用,這時你是不能使用retain release autorelease 操作的,因為ARC會自動為你做好這些操作,但是你需要在對象屬性上使用weak 和strong,其中strong就相當于retain屬性,而weak相當于assign。

2、只有一種情況你需要使用weak(默認是strong),就是為了避免retain cycles(就是父類中含有子類{父類retain了子類},子類中又調用了父類{子類又retain了父類},這樣都無法release)

3、聲明為weak的指針,指針指向的地址一旦被釋放,這些指針都將被賦值為nil。這樣的好處能有效的防止野指針。

七:ARC(Automatic Reference Counting):

1、就是代碼中自動加入了retain/release,原先需要手動添加的用來處理內存管理的引用計數的代碼可以自動地由編譯器完成了。

該機能在 iOS 5/ Mac OS X 10.7 開始導入,利用 Xcode4.2 以后可以使用該特性。

八:strong,weak,copy 具體用法:

1.具體一點:IBOutlet可以為weak,NSString為copy,Delegate一般為weak,其他的看情況。一般來說,類 “內部”的屬性設置為strong,類“外部”的屬性設置為weak。說到底就是一個歸屬權的問題。小心出現循環引用導致內存無法釋放。

2.不用ARC的話就會看到很多retian。

3.如果你寫了@synthesize abc = _abc;的話,系統自動幫你聲明了一個_abc的實例變量。

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

使用copy: 對NSString

使用retain: 對其他NSObject和其子類

注意:********************

assign:默認類型,setter方法直接賦值,而不進行retain操作

retain:setter方法對參數進行release舊值,再retain新值。

copy:setter方法進行Copy操作,與retain一樣

謝謝!!!

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容