@property中有哪些屬性關(guān)鍵字饶辙?
@property和@synthesize是成對(duì)出現(xiàn)的,可以自動(dòng)生成某個(gè)類成員變量的存取方法(getter和setter)迂苛。在Xcode4.5以及以后的版本中三热,@synthesize可以省略。
atomic與nonatomic
atomic:默認(rèn)是有該屬性的三幻,這個(gè)屬性是為了保證程序在多線程情況下康铭,編譯器會(huì)自動(dòng)生成一些互斥加鎖代碼,避免該變量的讀寫(xiě)不同步的問(wèn)題赌髓,提供多線程安全从藤。
nonatomic:如果該對(duì)象無(wú)需考慮多線程的情況,請(qǐng)加入這個(gè)屬性锁蠕,這樣會(huì)讓編譯器少生成一些互斥加鎖代碼夷野,禁止多線程,變量保護(hù)荣倾,提高性能和效率悯搔。
注:
atomic是Objc使用的一種線程保護(hù)技術(shù),基本上來(lái)講是防止在寫(xiě)未完成的時(shí)候另一個(gè)線程讀取舌仍,造成的數(shù)據(jù)錯(cuò)誤妒貌。而這種機(jī)制是非常耗費(fèi)系統(tǒng)資源的通危,所以iPhone這種小型設(shè)備上,如果沒(méi)有使用多線程間的通訊編程灌曙,那么nonatomic是一個(gè)非常好的選擇菊碟。而iOS開(kāi)發(fā)中,普遍使用nonatomic也是基于性能這一點(diǎn)在刺。readwrite與readonly
readwrite:這個(gè)屬性是默認(rèn)的情況逆害,會(huì)自動(dòng)為你生成存取器。
readonly:只生成getter蚣驼,不會(huì)生成setter方法魄幕。
注:
readwrite、readonly這兩個(gè)屬性的真正價(jià)值颖杏,不是提供成員變量的訪問(wèn)接口纯陨,而是控制成員變量的訪問(wèn)權(quán)限。strong與weak
strong:強(qiáng)引用留储,也是我們通常說(shuō)的引用队丝,其存亡直接決定了所指向?qū)ο蟮拇嫱觯?dāng)強(qiáng)引用指向了某個(gè)對(duì)象欲鹏,那便擁有了這個(gè)對(duì)象。如果不存在指向一個(gè)對(duì)象的引用臭墨,并且此對(duì)象不再使用赔嚎,則對(duì)象就會(huì)被從內(nèi)存中釋放掉。默認(rèn)所有實(shí)例變量和局部變量都是strong指針胧弛。
weak:弱引用,不決定對(duì)象的存亡尤误,只是單純的引用了某個(gè)對(duì)象但是并不擁有該對(duì)象。即使一個(gè)對(duì)象被持有無(wú)數(shù)個(gè)弱引用结缚,只要沒(méi)有強(qiáng)引用指向它损晤,那么還是會(huì)被清除。
注:
strong與retain功能相似红竭;weak與assign相似尤勋,只是當(dāng)對(duì)象消失后weak會(huì)自動(dòng)把指針置為nil,避免了野指針的產(chǎn)生,因而weak屬性就不需要在dealloc中置nil了茵宪。assign最冰、copy、retain
assgin:默認(rèn)類型稀火,setter方法直接賦值暖哨,不進(jìn)行任何的retain操作,不改變引用計(jì)數(shù)凰狞。一般用來(lái)處理基本數(shù)據(jù)類型篇裁。
retain:釋放舊的對(duì)象(release)沛慢,將舊對(duì)象的值賦給新對(duì)象,再令新對(duì)象引用計(jì)數(shù)為1达布⊥偶祝可理解為指針拷貝。
copy:與retain的流程一樣往枣,先對(duì)舊的值release,再copy出新的對(duì)象伐庭,引用計(jì)數(shù)為1,為了減少對(duì)上下文的依賴而引入的機(jī)制分冈』恚可以理解為內(nèi)容的拷貝,也就意味著內(nèi)容被copy后雕沉,內(nèi)存中會(huì)有兩個(gè)存儲(chǔ)空間存儲(chǔ)同樣的內(nèi)容集乔。
注:
使用assign: 對(duì)基礎(chǔ)數(shù)據(jù)類型 (NSInteger,CGFloat)和C數(shù)據(jù)類型(int, float, double, char, 等等)
使用copy: 對(duì)NSString
使用retain: 對(duì)其他NSObject和其子類
@synthesize和@dynamic分別有什么作用坡椒?
簡(jiǎn)單來(lái)講扰路,通過(guò)@synthesize指令告訴編譯器在編譯期間產(chǎn)生getter和setter方法。如果自定義getter和setter方法則會(huì)覆蓋編譯器幫我們生成的方法倔叼。
@dynamic指令告訴編譯器在編譯期間不自動(dòng)生成getter和setter方法汗唱,避免編譯期間產(chǎn)生警告。然后由自己實(shí)現(xiàn)存取方法或存取方法在運(yùn)行時(shí)動(dòng)態(tài)創(chuàng)建綁定丈攒。其主要作用就是用在NSManageObject對(duì)象的屬性聲明中哩罪,由于此類對(duì)象的屬性一般是從Core Data的屬性中生成的,Core Data框架會(huì)在程序運(yùn)行的時(shí)候?yàn)榇祟悓傩陨蒰etter和setter方法巡验。
ARC下际插,不顯式指定任何屬性關(guān)鍵字時(shí),默認(rèn)的關(guān)鍵字都有哪些显设?
atomic,readwrite,strong(對(duì)象),assgin(基本數(shù)據(jù)類型)框弛。
用@property聲明的NSString、NSArray捕捂、NSDictionary經(jīng)常使用copy關(guān)鍵字瑟枫,為什么?如果改用strong關(guān)鍵字指攒,可能造成什么問(wèn)題力奋?
這個(gè)問(wèn)題無(wú)非就是考察你對(duì)copy、strong這個(gè)兩個(gè)修飾符的理解幽七。簡(jiǎn)單來(lái)講,strong是強(qiáng)引用,仍舊指向同一個(gè)內(nèi)存地址景殷;copy是內(nèi)容拷貝,會(huì)另外開(kāi)辟一個(gè)內(nèi)存空間來(lái)存儲(chǔ)被拷貝的內(nèi)容,指針指向了一個(gè)不同的內(nèi)存地址猿挚。注意咐旧,copy返回的是一個(gè)不可變對(duì)象。如果用strong修飾可變對(duì)象绩蜻,那么這個(gè)對(duì)象就會(huì)有可能在不經(jīng)意間被修改铣墨,有時(shí)候這并不是我們的想要看到的,而用copy便不會(huì)有這種意外發(fā)生了办绝。
@synthesize合成實(shí)例變量的規(guī)則是什么伊约?假如property名為foo,存在一個(gè)名為_(kāi)foo的實(shí)例變量孕蝉,那么還會(huì)自動(dòng)合成新變量么屡律?
@synthesize表示由編譯器來(lái)自動(dòng)實(shí)現(xiàn)屬性的getter/setter方法,不需要你自己再手動(dòng)去實(shí)現(xiàn)降淮。默認(rèn)情況下超埋,不需要指定實(shí)例變量的名稱,編譯器會(huì)自動(dòng)生成一個(gè)屬性名前加“_”的實(shí)例變量佳鳖。當(dāng)然也可以在實(shí)現(xiàn)代碼里通過(guò)@synthesize語(yǔ)法來(lái)指定實(shí)例變量的名字霍殴。比如@synthesize foo = oof
(oof為你想要定義的名稱)。
如果property名為foo并且存在一個(gè)名為_(kāi)foo的實(shí)例變量系吩,編譯器便不會(huì)為我們自動(dòng)合成新變量了来庭。下面有一段代碼大家可以試一下幫助更好的理解。
@interface ViewController : UIViewController
@property (copy, nonatomic) NSString *testA;
@property (copy, nonatomic) NSString *testB;
@end
@interface ViewController ()
{
NSString *_testA;
NSString *_testB;
}
@end
@implementation ViewController
@synthesize testB = testBBBBB;
- (void)viewDidLoad {
[super viewDidLoad];
self.testA = @"1111";
self.testB = @"1111";
//輸出結(jié)果為:self.testA = 1111,_testA = 1111,self.testB = 1111,testBBBBB = 1111,_testB = (null)
NSLog(@"self.testA = %@,_testA = %@,self.testB = %@,testBBBBB = %@,_testB = %@",self.testA,_testA,self.testB,testBBBBB,_testB);
_testA = @"2222222";
_testB = @"2222222";
//輸出結(jié)果為:self.testA = 2222222,_testA = 2222222,self.testB = 1111,_testB = 2222222,testBBBBB = 1111
NSLog(@"self.testA = %@,_testA = %@,self.testB = %@,_testB = %@,testBBBBB = %@",self.testA,_testA,self.testB,_testB,testBBBBB);
testBBBBB = @"333333";
//輸出結(jié)果:self.testB = 333333,testBBBBB = 333333,_testB =2222222
NSLog(@"self.testB = %@,testBBBBB = %@,_testB =%@",self.testB,testBBBBB,_testB);
}
可加群一起交流共同學(xué)習(xí):801216530穿挨。