Objective-C高级编程:iOS与OS X多线程和内存管理
如何查看AutoreleasePoolPage類的情況(可查看某對象是否被自動release)
- 先聲明
extern void _objc_autoreleasePoolPrint();
- 再使用` _objc_autoreleasePoolPrint();
重要原則,在ARC中,編譯器會檢查方法名是否以alloc/new/copy/mutableCopy開始,如果<不是>則自動將返回值的對象註冊到autoreleasepool中
- 示例一(由於NSObject 是由alloc方法生成的所以不會自動添加到autoreleasepool中去,我們需要顯示加上__autoreleasing關鍵字,手動加入到pool中)
//在ARC下通過將對象賦值給__autoreleasing 的變量 相當於 非ARC下的autorelease方法,即對象註冊到autoreleasingPool
@autoreleasepool {//@autoreleasepool{} 相當於非ARC下NSAutoreleasePool類
id __autoreleasing obj = [[NSObject alloc]init];
}
MRC時的代碼:
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc]init];
id obj = [[NSObject alloc] init];
[obj autorelease];
[pool drain];
- 示例二:所以根據原則不顯示加上__autoreleasing關鍵字也可以,因為編譯器會根據方法名自動加入到autoreleasepool中
@autoreleasepool {
id __strong obj = [NSMutableArray array];
}
- 示例三:由於是alloc生成對象且return後,obj超出了其作用域,本應自動釋放,但該操作作為返回值,編譯器會自動加入到autoreleasepool中
+(id) array
{
id obj = [[NSMutableArray alloc]init];
return obj;
}
- 示例四:使用__weak修飾符的例子,編譯器要確保該對象存在,會自動加入到pool中
id __weak obj1 = obj0;
NSLog(@"class = %@",[obj1 class]);
//下面源碼同上面相同
id __weak obj1 = obj0;
id __autoreleasing tmp = obj1;
NSLog(@"class = %@",[tmp class]);
結論:id的指針和對象的指針會自動加上__autoreleasing修飾符
在ARC有效的情況下,必須遵守以下原則:
- 不能使用retian/release/retainCount/autorelease
- 不能使用NSAllocateObject/NSDeallocateObject
- mrc時alloc/new/copy/mutableCopy返回的對象為自己生成自己持有,這在arc時也是一樣,不過要加上一條->init實例方法,此方法必須要返回對象,也不會註冊到autoreleasepoll上,基本只是對alloc返回的對象進行初始化並返回該對象
id obj = [[NSObject alloc]init]
必須遵守駝峰命名法,例如:
- (id) initWithObject:(id)obj;
- 不要顯式調用dealloc
- 使用@autoreleasepool 塊替代NSAutoreleasePool
- 不能使用NSZone
- 對象型變量不能作為C語言結構體成員
- 顯式轉換 id 和 void*
id obj = [[NSObject alloc ] init];
void *p = (__bridge void *)obj;
id o = (__bridge id)p;//但這樣轉換很不安全!不會將指針清nil
還有另外兩種__bridge_retained與retian類似,__bridge_transfer與release類似
//ARC
void *p = (__bridge_retained void*)[[NSObject alloc]init];
NSLog(@"class = %@",[(__bridge id)p class]);
(void)(__bridge_transfer id)p;
//MRC
id p = [[NSObject alloc]init];
NSLog(@"class = %@",[p class]);
[p release]