注:本文在每段代碼中首段標有 "http://ARC" 意為使用ARC的代碼, "http://MRC"則為MRC的代碼
</br>
//ARC
+ (id) array {
return [[NSMutableArray alloc] init];
}
上面是NSMutableArray的類方法,?這樣返回在ARC里面沒有任何問題,因為編譯器會自動替你完成工作。
現在假設我們在MRC的環境里:
//MRC
+ (id) array {
return [[NSMutableArray alloc] init];
}
這就有問題了,直接返回之后你沒有辦法釋放alloc之后創建的對象(造成內存泄露)。對了,你沒有一個指針對象可以訪問到它,來做釋放操作。你也許想到這么做:
//MRC
+ (id) array {
NSMutableArray *array = [[NSMutableArray alloc] init];
return array;
}
但這么做也不對啊。雖說有一個指針可以訪問并釋放它了,但它是在return前釋放也不是,在return后釋放就更不對了。
所有我們需要一種可以返回了這個對象,并在晚點再釋放的機制——自動釋放池(autoreleasepool)。
把返回的對象注冊到autoreleasepool里面可以延遲釋放,只要對其調用autorelease:
//MRC
+ (id) array {
return [[[NSMutableArray alloc] init] autorelease];
}
上面的代碼等價于在ARC時:
//ARC
+ (id) array {
return [[NSMutableArray alloc] init];
}
autorelease的方法ARC已經幫你調用了,只是你看不到。
</br>
再回來看:
//ARC
id obj = [NSMutableArray array];
//等價于
id __strong obj = [NSMutableArray array];
在ARC里,對象默認是strong的,所以obj要持有對象,為了不讓返回的對象自動釋放,所以?會自動調用一次retain。相當于MRC如下代碼:
//MRC
id temp = [NSMutableArray array];
id obj = [temp retain];
綜上,ARC為我們自動調用了autorelease和retain這兩個方法。
如果我們使用MRC實現,是否需要這么麻煩呢?你重翻看上面的代碼就?發現,MRC不調用autorelease和retain方法,也實現了同樣的事情。
所以調用跟不調用autorelease和retain這兩個方法時的情況是一樣的。ARC白白調用了2次多余的方法,又沒有多干活,肯定影響性能。ARC這么傻,我為什么還要用它?ARC肯定采取了措施進行了優化。
因為直接將二者刪去這種處理方式,又要考慮到“向后兼容性”問題,以兼容那些不使用ARC的代碼。
所以ARC就使用objc_autoreleaseReturnValue和
objc_retainAutoreleasedReturnValue這兩個函數對這些多余的操作進行優化。
</br>
在使用非alloc/new/copy/mutableCopy等開頭生成對象的方法,都會調用這兩個方法,比如array類方法會變成下面這樣,objc_autoreleaseReturnValue函數替代了autorelease方法:
+ (id) array {
NSMutableArray *array = [[NSMutableArray alloc] init];
objc_autoreleaseReturnValue(array);
}
當在外部調用array方法賦值時方法是這樣,
objc_retainAutoreleasedReturnValue函數替代了retain方法:
id temp = [NSMutable array];
id obj = objc_retainAutoreleasedReturnValue(temp);
如上代碼所示,會先調用objc_autoreleaseReturnValue?函數,它的作用是檢視稍后執行的代碼是否會執行retain方法,如果有,會把某個專用于檢測的變量,或者說數據結構的標志位置位,并直接返回對象(上例array),不執行autorelease方法;否則,對對象(上例array)執行autorelease方法。 objc_retainAutoreleasedReturnValue函數則是訪問專用于檢測的變量的標志位,看其是否有置位,如果有,直接返回對象;否則,對對象執行retain方法。查看一個標志位,一般都要比調用autorelease和retain來得快吧?
id objc_autoreleaseReturnValue(id object) {
if ( //調用者將會執行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函數的else分支上要執行一次retain。希望可以交流一下。