注:本文在每段代碼中首段標(biāo)有 "http://ARC" 意為使用ARC的代碼尾序, "http://MRC"則為MRC的代碼
</br>
//ARC
+ (id) array {
return [[NSMutableArray alloc] init];
}
上面是NSMutableArray的類方法扩劝,?這樣返回在ARC里面沒有任何問題中狂,因?yàn)榫幾g器會自動替你完成工作手素。
現(xiàn)在假設(shè)我們在MRC的環(huán)境里:
//MRC
+ (id) array {
return [[NSMutableArray alloc] init];
}
這就有問題了祖搓,直接返回之后你沒有辦法釋放alloc之后創(chuàng)建的對象(造成內(nèi)存泄露)佑力。對了底桂,你沒有一個指針對象可以訪問到它植袍,來做釋放操作。你也許想到這么做:
//MRC
+ (id) array {
NSMutableArray *array = [[NSMutableArray alloc] init];
return array;
}
但這么做也不對啊籽懦。雖說有一個指針可以訪問并釋放它了于个,但它是在return前釋放也不是,在return后釋放就更不對了暮顺。
所有我們需要一種可以返回了這個對象厅篓,并在晚點(diǎn)再釋放的機(jī)制——自動釋放池(autoreleasepool)秀存。
把返回的對象注冊到autoreleasepool里面可以延遲釋放,只要對其調(diào)用autorelease:
//MRC
+ (id) array {
return [[[NSMutableArray alloc] init] autorelease];
}
上面的代碼等價于在ARC時:
//ARC
+ (id) array {
return [[NSMutableArray alloc] init];
}
autorelease的方法ARC已經(jīng)幫你調(diào)用了羽氮,只是你看不到或链。
</br>
再回來看:
//ARC
id obj = [NSMutableArray array];
//等價于
id __strong obj = [NSMutableArray array];
在ARC里,對象默認(rèn)是strong的档押,所以obj要持有對象澳盐,為了不讓返回的對象自動釋放,所以?會自動調(diào)用一次retain汇荐。相當(dāng)于MRC如下代碼:
//MRC
id temp = [NSMutableArray array];
id obj = [temp retain];
綜上洞就,ARC為我們自動調(diào)用了autorelease和retain這兩個方法。
如果我們使用MRC實(shí)現(xiàn)掀淘,是否需要這么麻煩呢旬蟋?你重翻看上面的代碼就?發(fā)現(xiàn),MRC不調(diào)用autorelease和retain方法革娄,也實(shí)現(xiàn)了同樣的事情倾贰。
所以調(diào)用跟不調(diào)用autorelease和retain這兩個方法時的情況是一樣的。ARC白白調(diào)用了2次多余的方法拦惋,又沒有多干活匆浙,肯定影響性能。ARC這么傻厕妖,我為什么還要用它首尼?ARC肯定采取了措施進(jìn)行了優(yōu)化。
因?yàn)橹苯訉⒍邉h去這種處理方式言秸,又要考慮到“向后兼容性”問題软能,以兼容那些不使用ARC的代碼。
所以ARC就使用objc_autoreleaseReturnValue和
objc_retainAutoreleasedReturnValue這兩個函數(shù)對這些多余的操作進(jìn)行優(yōu)化举畸。
</br>
在使用非alloc/new/copy/mutableCopy等開頭生成對象的方法查排,都會調(diào)用這兩個方法,比如array類方法會變成下面這樣抄沮,objc_autoreleaseReturnValue函數(shù)替代了autorelease方法:
+ (id) array {
NSMutableArray *array = [[NSMutableArray alloc] init];
objc_autoreleaseReturnValue(array);
}
當(dāng)在外部調(diào)用array方法賦值時方法是這樣跋核,
objc_retainAutoreleasedReturnValue函數(shù)替代了retain方法:
id temp = [NSMutable array];
id obj = objc_retainAutoreleasedReturnValue(temp);
如上代碼所示,會先調(diào)用objc_autoreleaseReturnValue?函數(shù)叛买,它的作用是檢視稍后執(zhí)行的代碼是否會執(zhí)行retain方法砂代,如果有,會把某個專用于檢測的變量率挣,或者說數(shù)據(jù)結(jié)構(gòu)的標(biāo)志位置位泊藕,并直接返回對象(上例array),不執(zhí)行autorelease方法;否則娃圆,對對象(上例array)執(zhí)行autorelease方法。 objc_retainAutoreleasedReturnValue函數(shù)則是訪問專用于檢測的變量的標(biāo)志位蛾茉,看其是否有置位讼呢,如果有,直接返回對象谦炬;否則悦屏,對對象執(zhí)行retain方法。查看一個標(biāo)志位键思,一般都要比調(diào)用autorelease和retain來得快吧础爬?
id objc_autoreleaseReturnValue(id object) {
if ( //調(diào)用者將會執(zhí)行retain ) {
set_flag(object);
return object;
} else {
return [object autorelease];
}
}
id objc_retainAutoreleasedReturnValue(id object) {
if (get_flag(object)) {
clear_flag(object);
return object;
} else {
return [object retain];
}
}
以上2個方法摘自《Effective Objective-C 2.0》p126。筆者也尚未搞明為何objc_retainAutoreleasedReturnValue函數(shù)的else分支上要執(zhí)行一次retain吼鳞。希望可以交流一下看蚜。