在文章開始之前我想讓大家先思考兩個問題:
weak和assign的區(qū)別?
1楼咳、修飾變量的區(qū)別
weak 只可以修飾對象辟汰,如果修飾基本數(shù)據(jù)類型,則編譯器會報錯岛宦。
assign 既可以修飾對象台丛,也可以修飾基本數(shù)據(jù)類型。
2恋博、是否產(chǎn)生野指針區(qū)別:
(1)weak 不會產(chǎn)生野指針問題齐佳,因?yàn)?weak 修飾的對象釋放后(引用計數(shù)器值為0),指針會自動被置nil债沮,之后再向該對象發(fā)消息也不會崩潰炼吴,所以 weak 是安全的。
(2)assign 如果修飾對象疫衩,會產(chǎn)生野指針問題硅蹦,修飾的對象釋放后,指針不會自動被置空闷煤,此時再向?qū)ο蟀l(fā)消息則會崩潰童芹;如果修飾基本數(shù)據(jù)類型則是安全的。
總結(jié):
(1)assign 適用于基本數(shù)據(jù)類型如 int鲤拿,float假褪,struct 等值類型;
(2)weak 適用于 delegate 和 block 等引用類型近顷,不會導(dǎo)致野指針問題生音,也不會循環(huán)引用宁否,非常安全。
strong和copy的區(qū)別缀遍?
我們都知道
strong
和copy
修飾對象時都是強(qiáng)引用慕匠,持有對象,而且引用計數(shù)器都會加一域醇,那么他們二者之間到底有什么具體的區(qū)別呢台谊?用
@property
聲明的NSString(或NSArray,NSDictionary)
經(jīng)常使用copy關(guān)鍵字譬挚,為什么锅铅?如果改用strong關(guān)鍵字,可能造成什么問題殴瘦?
在具體開始之前我們先來看看下面這段代碼:
@interface ViewController ()
@property (nonatomic,strong) NSArray * array;
@property (nonatomic,strong) NSMutableArray * muArrayS;
@property (nonatomic,copy) NSMutableArray * muArrayC;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
NSMutableArray * muArray=[NSMutableArray array];
self.muArrayC=muArray;
self.muArrayS=muArray;
NSLog(@"muArrayC:%@",[self.muArrayC class]);
NSLog(@"muArrayS:%@",[self.muArrayS class]);
[self.muArrayC removeAllObjects];
[self.muArrayS removeAllObjects];
}
執(zhí)行結(jié)果:
2017-03-02 23:31:01.656 joke[3504:207330] muArrayC:__NSArray0
2017-03-02 23:31:01.656 joke[3504:207330] muArrayS:__NSArrayM
2017-03-02 23:31:01.656 joke[3504:207330] -[__NSArray0 removeAllObjects]:
unrecognized selector sent to instance 0x608000016910
2017-03-02 23:31:01.659 joke[3504:207330] *** Terminating app due to uncaught
exception 'NSInvalidArgumentException', reason: '-[__NSArray0 removeAllObjects]:
unrecognized selector sent to instance 0x608000016910'
What狠角?什么原因。蚪腋。。姨蟋。不著急屉凯,暫且往下看
strong
首先我們來說說這個我認(rèn)為比較好理解的strong
,它其實(shí)是一個非常簡單的屬性修飾符眼溶,用strong
修飾的屬性在進(jìn)行賦值操作的時候悠砚,右邊數(shù)據(jù)是什么類型那么左邊就是什么類型,也就是說誰把對象給了它堂飞,則它就指向哪個對象灌旧,并且這個屬性如果你不主動把它清空,它就會一直存在直到所有引用它的對象都被釋放時绰筛,它才會釋放枢泰。
@interface ViewController ()
@property (nonatomic,strong) NSArray * array;
@end
@implementation ViewController
- (void)viewDidLoad {
[super viewDidLoad];
NSMutableArray *muArray = [NSMutableArray array];
self.array = muArray;
NSLog(@"%@",[self.array class]);
}
執(zhí)行結(jié)果:
[3176:181971] __NSArrayM
從上面我們可以看到用strong
修飾的NSArry
,當(dāng)外界傳遞進(jìn)來一個NSMutableArray
的時候铝噩,此時NSArray
對象就指向了一個可變數(shù)組了衡蚂。
copy
我們來看一段代碼:
NSString *string = @"The Great China";
NSString *copyString = [string copy];// 不創(chuàng)建出新對象,指針與源對象相同
NSMutableString *mutableCopyString = [string mutableCopy];// //創(chuàng)建出新對象骏庸,指針與源對象不同
NSLog(@"string = %p copyString = %p mutableCopyString = %p", string, copyString, mutableCopyString);
執(zhí)行結(jié)果:
string = 0x10b1a8068 copyString = 0x10b1a8068 mutableCopyString = 0x608000073c00
-copy, always returns their immutable counterparts. Thus, when an NSMutableArray is sent -copy, it returns an NSArray containing the same objects.
使用 copy
的目的是為了讓本對象的屬性不受外界影響毛甲,使用 copy
無論外界給我傳入一個可變對象還是不可變對象,我本身持有的就是一個不可變的副本
所以copy出來的仍然是不可變字符具被!當(dāng)我們調(diào)用NSMutableArray的方法時玻募,程序就會崩潰:
總結(jié)
到這里,想必大家心里已經(jīng)對文章一開始的兩個問題有了答案一姿。
- 因?yàn)楦割愔羔樋梢灾赶蜃宇悓ο?使用
copy
的目的是為了讓本對象的屬性不受外界影響七咧,使用copy
無論給我傳入是一個可變對象還是不可變對象跃惫,我本身持有的就是一個不可變的副本. - 如果我們使用的是
strong
,那么這個屬性就有可能指向一個可變對象坑雅,如果這個可變對象在外部被修改了辈挂,那么會影響該屬性。
總結(jié):
- strong對應(yīng)的setter方法裹粤,是將_property先release(_property release)终蒂,然后將參數(shù)retain(property retain),最后是_property = property遥诉。
- copy對應(yīng)的setter方法拇泣,是將_property先release(_property release),然后拷貝參數(shù)內(nèi)容(property copy)矮锈,創(chuàng)建一塊新的內(nèi)存地址霉翔,最后_property = property。
當(dāng)屬性類型為NSString或者NSArray
等對象時苞笨,經(jīng)常用Copy來保護(hù)其封裝性债朵,因?yàn)閭鬟f給設(shè)置方法的新值有可能指向一個NSMutable
可變類的實(shí)例,所以:
當(dāng)修飾不可變類型的屬性時瀑凝,如NSArray序芦、NSDictionary、NSString粤咪,
用copy谚中。
當(dāng)修飾可變類型的屬性時,如NSMutableArray寥枝、NSMutableDictionary宪塔、NSMutableString,
用strong
囊拜。
最后分享一個陽神出的面試題給大家某筐,看看下面這四種寫法的區(qū)別?
@property(nonatomic,strong)NSArray * arrry0;
@property(nonatomic,copy)NSArray * arrry1;
@property(nonatomic,copy)NSMutableArray * arrry3;
@property(nonatomic,strong)NSMutableArray * arrry4;
深淺拷貝的問題
淺拷貝就是對內(nèi)存地址的復(fù)制艾疟,讓目標(biāo)對象指針和源對象指針指向同一片內(nèi)存空間来吩。如下圖所示:
深拷貝讓目標(biāo)對象指針和源對象指針指向兩片內(nèi)容相同的內(nèi)存空間。如下圖所示:
深拷貝和淺拷貝的區(qū)別:
- 深拷貝開辟了新的內(nèi)存空間蔽莱,而淺拷貝則沒有
- 深拷貝不會影響對象的引用計數(shù)弟疆,而淺拷貝則會影響被拷貝對象的引用計數(shù)。
@property (nonatomic,copy) NSString * stringCopy;
NSMutableString *muString=[NSMutableString stringWithFormat:@"China"];
self.stringCopy = muString;
NSLog(@"muString:%p copyString:%p",muString,self.stringCopy);
執(zhí)行結(jié)果:
muString:0x6000002617c0 copyString:0xa0000616e6968435
查看內(nèi)存盗冷,會發(fā)現(xiàn) muString怠苔、stringCopy 內(nèi)存地址都不一樣,說明此時都是做內(nèi)容拷貝仪糖、深拷貝柑司。即使你進(jìn)行如下操作:
[muString appendString:@"Great!"];
stringCopy
的值也不會因此改變迫肖,但是如果stringCopy
不使用 copy,
修飾 攒驰,stringCopy
的值就會被改變蟆湖。
總結(jié):
- 在非集合類對象中進(jìn)行
copy
操作,是指針復(fù)制玻粪,mutableCopy
操作是內(nèi)容復(fù)制隅津; - 對集合對象進(jìn)行
copy
和mutableCopy
都是內(nèi)容復(fù)制。
注意:上述原則對其他對象劲室,如NSArray伦仍、NSMutableArray 、NSDictionary很洋、NSMutableDictionary一樣適用
總結(jié):
- copy方法返回的都是不可變對象充蓝。
- mutableCopy都是深拷貝。
- 可變對象的copy是深拷貝喉磁,不可變對象的copy是淺拷貝谓苟。
[immutableObject copy] // 淺復(fù)制
[immutableObject mutableCopy] //深復(fù)制
[mutableObject copy] //深復(fù)制
[mutableObject mutableCopy] //深復(fù)制
思考:
如何讓自己的類用 copy 修飾符?如何重寫帶 copy 關(guān)鍵字的 setter协怒?
若想令自己所寫的對象具有拷貝功能娜谊,則需實(shí)現(xiàn) NSCopying 協(xié)議。如果自定義的對象分為可變版本與不可變版本斤讥,那么就要同時實(shí)現(xiàn) NSCopying 與 NSMutableCopying 協(xié)議。
具體步驟:
1湾趾、 需聲明該類遵從 NSCopying 協(xié)議
2芭商、 實(shí)現(xiàn) NSCopying 協(xié)議。該協(xié)議只有一個方法:
- (id)copyWithZone:(NSZone *)zone;