//聯(lián)系人:石虎QQ: 1224614774昵稱:嗡嘛呢叭咪哄
OC?關(guān)鍵字:copy,assign,strong,retain,weak,readonly,readwrite,nonatomic,atomic,unsafe_unretained的使用與區(qū)別
一、關(guān)鍵字的概念:
ARC-自動(dòng)醫(yī)用技術(shù)
ARC不是垃圾回收,而是編譯器自動(dòng)插入代碼來(lái)減少程序員的代碼輸入和失誤。
同時(shí)比垃圾和效率要高,因?yàn)槠洳挥绊戇\(yùn)行時(shí)間,相當(dāng)于自己管理內(nèi)存。
總是通過(guò)屬性來(lái)管理實(shí)例變量(init/dealloc除外),在dealloc中釋放所有屬性。
dealloc中會(huì)自動(dòng)加入釋放實(shí)例變量的代碼,因此不必要手段增加釋放實(shí)例變量的代碼。不需要手動(dòng)調(diào)用[super? dealloc]
不要調(diào)用retain,release,autorelease,編譯器會(huì)自動(dòng)插入相關(guān)代碼。
注意命名方式,不要以copyXXX方式命名不想進(jìn)行retain的方法,編譯器會(huì)根據(jù)方法名自動(dòng)retain。
C語(yǔ)言結(jié)構(gòu)體中不要有對(duì)象指針
id和void*只能通過(guò)橋接轉(zhuǎn)換來(lái)進(jìn)行轉(zhuǎn)換
不要使用NSAutoreleasePool,而是使用@autoreleasepool{}代碼塊。
轉(zhuǎn)換ARC代碼:Edit->Refactor->Convert? to Objective-C ARC
retain cycle
循環(huán)保留
delegate和block是產(chǎn)生retain? cycle的主要原因
dealloc
移除觀察者observers
注銷通知notification
設(shè)置非weak的delegate為nil??? 取消timer
二、關(guān)鍵字含義:
copy
復(fù)制內(nèi)容(深復(fù)制),如果調(diào)用copy的是數(shù)組,則為指針復(fù)制(淺復(fù)制),僅僅復(fù)制子元素的指針。
@property? (nonatomic,copy)NSString? *title;
@property (nonatomic, copy) NSMutableArray *myArray;//not recommended
@property (nonatomic, copy) SomeBlockType someBlock;
assign
對(duì)基礎(chǔ)數(shù)據(jù)類型(NSInteger,CGFloat)和C數(shù)據(jù)類型(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
相當(dāng)于retain。
Strong在ARC環(huán)境為默認(rèn)屬性類型。
@property? (nonatomic,readwrite,strong)NSString *title;
@property (strong, nonatomic) UIViewController *viewController;
@property (nonatomic,? strong) id? childObject;
retain
NSObject及其子類。
Release舊值,retain新值。
Retain是指針復(fù)制(淺復(fù)制),引用計(jì)數(shù)加1,而不會(huì)導(dǎo)致內(nèi)容被復(fù)制。
@property? (nonatomic, retain)UIColor *myColor;
weak
取代之前的assign,對(duì)象銷毀之后會(huì)自動(dòng)置為nil,防止野指針。
Assign不能自動(dòng)置為nil,需要手動(dòng)置為nil。
Delegate基本總是使用weak,以防止循環(huán)引用。特殊情況是,希望在dealloc中調(diào)用delegate的某些方法進(jìn)行釋放,此時(shí)如果使用weak將引起異常,因?yàn)榇藭r(shí)已經(jīng)是nil了,那么采用assign更為合適。
@property? (weak, nonatomic) IBOutlet UIButton *myButton;//處于最頂層的IBOutlet應(yīng)該為strong
@property (nonatomic,? weak) id? parentObject;
@property(nonatomic,readwrite,weak) id? delegate;
@property (nonatomic,? weak) NSObject? *delegate;
readonly
此標(biāo)記說(shuō)明屬性是只讀的,默認(rèn)的標(biāo)記是讀寫,如果你指定了只讀,在@implementation中只需要一個(gè)讀取器。或者如果你使用@synthesize關(guān)鍵字,也是有讀取器方法被解析。而且如果你試圖使用點(diǎn)操作符為屬性賦值,你將得到一個(gè)編譯錯(cuò)誤。
readwrite
此標(biāo)記說(shuō)明屬性會(huì)被當(dāng)成讀寫的,這也是默認(rèn)屬性。設(shè)置器和讀取器都需要在@implementation中實(shí)現(xiàn)。如果使用@synthesize關(guān)鍵字,讀取器和設(shè)置器都會(huì)被解析。
unsafe_unretained
unretained且unsafe,由于是unretained所以與weak有點(diǎn)類似,但是它是unsafe的.
@property(nonatomic,unsafe_unretained)Book *book1;
unsafe_unretainedid safeSelf = self;
三.關(guān)鍵字使用區(qū)別:
1:copy與retain:
1、copy其實(shí)是建立了一個(gè)相同的對(duì)象,而retain不是;
2、copy是內(nèi)容拷貝,retain是指針拷貝;
3、copy是內(nèi)容的拷貝 ,對(duì)于像NSString,的確是這樣,但是如果copy的是一個(gè)NSArray呢?這時(shí)只是copy了指向array中相對(duì)應(yīng)元素的指針.這便是所謂的"淺復(fù)制".
4、copy的情況:NSString *newPt = [pt copy];
此時(shí)會(huì)在堆上重新開(kāi)辟一段內(nèi)存存放@"abc" 比如0X1122 內(nèi)容為@"abc 同時(shí)會(huì)在棧上為newPt分配空間 比如地址:0Xaacc 內(nèi)容為0X1122 因此retainCount增加1供newPt來(lái)管理0X1122這段內(nèi)存;
二:assign與retain:
1、assign: 簡(jiǎn)單賦值,不更改索引計(jì)數(shù);
2、assign的情況:NSString *newPt = [pt assing];
此時(shí)newPt和pt完全相同 地址都是0Xaaaa 內(nèi)容為0X1111 即newPt只是pt的別名,對(duì)任何一個(gè)操作就等于對(duì)另一個(gè)操作, 因此retainCount不需要增加;
3、assign就是直接賦值;
4、retain使用了引用計(jì)數(shù),retain引起引用計(jì)數(shù)加1, release引起引用計(jì)數(shù)減1,當(dāng)引用計(jì)數(shù)為0時(shí),dealloc函數(shù)被調(diào)用,內(nèi)存被回收;
5、retain的情況:NSString *newPt = [pt retain];
此時(shí)newPt的地址不再為0Xaaaa,可能為0Xaabb 但是內(nèi)容依然為0X1111。 因此newPt 和 pt 都可以管理"abc"所在的內(nèi)存,因此 retainCount需要增加1;
三:readonly:
readonly:只產(chǎn)生簡(jiǎn)單的getter,沒(méi)有setter。
四:readwrite:
readwrite:同時(shí)產(chǎn)生setter\getter方法
五nonatomic,atomic:
:1、非原子性訪問(wèn),對(duì)屬性賦值的時(shí)候不加鎖,多線程并發(fā)訪問(wèn)會(huì)提高性能。如果不加此屬性,則默認(rèn)是兩個(gè)訪問(wèn)方法都為原子型事務(wù)訪問(wèn);
weak and strong property (強(qiáng)引用和弱引用的區(qū)別):
:2,置成員變量的@property屬性時(shí),默認(rèn)為atomic,提供多線程安全。在多線程環(huán)境下,原子操作是必要的,否則有可能引起錯(cuò)誤的結(jié)果
atomic的意思就是setter/getter這個(gè)函數(shù),是一個(gè)原語(yǔ)操作。如果有多個(gè)線程同時(shí)調(diào)用setter的話,不會(huì)出現(xiàn)某一個(gè)線程執(zhí)行完setter全部語(yǔ)句之前,另一個(gè)線程開(kāi)始執(zhí)行setter情況,相當(dāng)于函數(shù)頭尾加了鎖一樣,可以保證數(shù)據(jù)的完整性。nonatomic不保證setter/getter的原語(yǔ)行,所以你可能會(huì)取到不完整的東西。因此,在多線程的環(huán)境下原子操作是非常必要的,否則有可能會(huì)引起錯(cuò)誤的結(jié)果。
比如setter函數(shù)里面改變兩個(gè)成員變量,如果你用nonatomic的話,getter可能會(huì)取到只更改了其中一個(gè)變量時(shí)候的狀態(tài),這樣取到的東西會(huì)有問(wèn)題,就是不完整的。當(dāng)然如果不需要多線程支持的話,用nonatomic就夠了,因?yàn)椴簧婕暗骄€程鎖的操作,所以它執(zhí)行率相對(duì)快些。
六:weak 和 strong
1、weak 和 strong 屬性只有在你打開(kāi)ARC時(shí)才會(huì)被要求使用,這時(shí)你是不能使用retain release autorelease 操作的,因?yàn)锳RC會(huì)自動(dòng)為你做好這些操作,但是你需要在對(duì)象屬性上使用weak 和strong,其中strong就相當(dāng)于retain屬性,而weak相當(dāng)于assign。
2、只有一種情況你需要使用weak(默認(rèn)是strong),就是為了避免retain cycles(就是父類中含有子類{父類retain了子類},子類中又調(diào)用了父類{子類又retain了父類},這樣都無(wú)法release)
3、聲明為weak的指針,指針指向的地址一旦被釋放,這些指針都將被賦值為nil。這樣的好處能有效的防止野指針。
七:ARC(Automatic Reference Counting):
1、就是代碼中自動(dòng)加入了retain/release,原先需要手動(dòng)添加的用來(lái)處理內(nèi)存管理的引用計(jì)數(shù)的代碼可以自動(dòng)地由編譯器完成了。
該機(jī)能在 iOS 5/ Mac OS X 10.7 開(kāi)始導(dǎo)入,利用 Xcode4.2 以后可以使用該特性。
八:strong,weak,copy 具體用法:
1.具體一點(diǎn):IBOutlet可以為weak,NSString為copy,Delegate一般為weak,其他的看情況。一般來(lái)說(shuō),類 “內(nèi)部”的屬性設(shè)置為strong,類“外部”的屬性設(shè)置為weak。說(shuō)到底就是一個(gè)歸屬權(quán)的問(wèn)題。小心出現(xiàn)循環(huán)引用導(dǎo)致內(nèi)存無(wú)法釋放。
2.不用ARC的話就會(huì)看到很多retian。
3.如果你寫了@synthesize abc = _abc;的話,系統(tǒng)自動(dòng)幫你聲明了一個(gè)_abc的實(shí)例變量。
使用assign: 對(duì)基礎(chǔ)數(shù)據(jù)類型 (NSInteger)和C數(shù)據(jù)類型(int, float, double, char,等)
使用copy: 對(duì)NSString
使用retain: 對(duì)其他NSObject和其子類
注意:********************
assign:默認(rèn)類型,setter方法直接賦值,而不進(jìn)行retain操作
retain:setter方法對(duì)參數(shù)進(jìn)行release舊值,再retain新值。
copy:setter方法進(jìn)行Copy操作,與retain一樣
謝謝!!!