【iOS】OC高级编程 iOS多线程与内存管理阅读笔记——自动引用计数(二) ...

打印 上一主题 下一主题

主题 987|帖子 987|积分 2961

alloc/retain/release/dealloc实现

包罗NSObject类的Foundation框架的源代码没有公开,不外Foundation框架使用的Core Foundation框架的源代码以及通过调用NSObject类进行内存管理部分的源代码是公开的。但是没有NSObject类的源代码,就很难了解NSObject类的内部实现细节。为此,我们使用开源软件GNUstep来说明,
GNUstep是Cocoa框架的互换框架,也就是说在使用者看来,两者的行为和实现方式是一样的,或者说非常相似。
GNUstep源代码中NSObject类的alloc类方法是这样的:
  1. id obj = [NSOject alloc];
复制代码
在NSObject.m源代码中的实现如下:
  1. + (id)alloc {
  2.     return [self allocWithZone:NSDefaultMallocZone()];
  3. }
  4. + (id)allocWithZone:(NSZone*)z {
  5.     return NSAllocateObject(self, 0, z);
  6. }
复制代码
通过allocWithZone:类方法调用NSAllocateObject函数分配了对象,下面来看看NSAllocateObject函数:
  1. struct obj_layout {
  2.     NSUInteger retained;
  3. }
  4. inline id
  5. NSAllocateObject (Class aClass, NSUInteger extraBytes, NSZone *zone) {
  6.     int size = 计算容纳对象所需内存大小;
  7.     id new = NSZoneMalloc (zone, size);
  8.     memset (new, 0, size);
  9.     new = (id) & ((struct obj_layout *)new)[1];
  10. }
复制代码
NSAllocateObject函数通过调用NSZoneMalloc函数来分配存放对象所需的内存空间,之后将该内存空间置0,末了返回作为对象而使用的指针。
   NSDefaultMallocZone、NSZoneMalloc等名称中包罗的NSZone,是为防止内存碎片化而引入的布局。对内存分配的区域自己进行多重化管理,根据使用对象的目标、对象的大小分配内存,从而进步内存管理的服从。
  但是现在的运行时体系只是简朴地忽略了区域的概念,运行时体系中的内存管理自己已极具服从,使用区域来管理内存反而会引起内存使用服从低下以及源代码复杂化等标题。

  以下是去掉NSZone后简化了的源代码:
  1. struct obj_layout {
  2.     NSUInteger retained;
  3. };
  4. + (id)alloc {
  5.     int size = sizeof (struct obj_layout) + 对象大小;
  6.     struct obj_layout *p = (struct obj_layout*)calloc(1,size);
  7.     return (id)(p+1);
  8. }
复制代码
alloc类方法用struct obj_layout中的retained整数来保存引用计数,并将其写入对象内存头部,该对象内存块全部置零后返回。

对象的引用计数可以通过retainCount实例方法取得。
  1. id obj = [[NSObject alloc] init];
  2. NSLog(@"retainCount = %d", [obj retainCount]);
  3. //显示retainCount = 1
复制代码
执行alloc后对象的retainCount是"1"。可以通过GNUstep的源代码来确认
  1. - (NSUInteger)retainCount {
  2.     return NSExtraRefCount(self) + 1;
  3. }
  4. inline NSUInteger
  5. NSExtraRefCount (id anObject) {
  6.     return ((struct obj_layout *)anObject)[-1].retained;
  7. }
复制代码
由对象寻址找到对象内存头部,从而访问其中的retained变量。

分配时全部置0,所以retained为0。再由NSExtraRefCount(self) + 1可得出,retainCount为1。不难推测出:retain方法使retained变量加1,而release方法使retained变量减1。
  1. [obj retain];
复制代码
下面是retain的源代码
  1. - (id) retain {
  2.     NSIncrementExtraRefCount(self);
  3.     return self;
  4. }
  5. inline void
  6. NSIncrementExtraRefCount (id anObject) {
  7.     if (((struct obj_layout *)anObject)[-1].retained == UINT_MAX - 1)
  8.           [NSException raise: NSInternalInconsistencyException                                      format:@"NSIncrementExtraRefCount()asked to increment too far"];
  9.     ((struct obj_layout *)anObject) [-1].retained++;
  10. }
复制代码
虽然有当retained变量超出最大值时发生异常的代码,但实际上只运利用retained变量加一的retained++代码/同样,release实例方法进行retained--并在该引用计数变量为0时作出处理。下面通过源代码来确认。
  1. [obj release];
复制代码
  1. - (void)release {
  2.       if (NSDecrementExtraRefCountWasZero(self))
  3.       [self dealloc];
  4. }
  5. BOOL
  6. NSDecrementExtraRefCpuntWasZero (id anObject) {
  7.     if (((struct obj_layout *)anObject)[-1].retained == 0) {
  8.       return YES;
  9.     } else {
  10.     ((struct obj_layout *)anObject)[-1].retained--;
  11.     return NO;
  12.     }
  13. }
复制代码
当retained变量大于0时减1,等于0时调用dealloc实例方法,废弃对象。以下是废弃对象时所调用的dealloc实例方法的实现。
  1. - (void)dealloc {
  2.     NSDeallocateObject(self);
  3. }
  4. inline void
  5. NSDeallocateObject(id anObject) {
  6.     struct obj_layout *o = &((struct obj_layout *)anObject)[-1];
  7.     free(o);
  8. }
复制代码
上述代码仅废弃由alloc分配的内存块。
以上是alloc/retain/release/dealloc在GNUstep中的实现。总结如下:


  • OC对象中存有引用计数这一整数值
  • 调用alloc或是retain方法,引用计数加一
  • 调用release后,引用计数减一
  • 引用计数值为零时,调用dealloc方法废弃对象
苹果的实现

接下来看看苹果是怎么实现内存管理和引用计数的。由于NSObject类的源代码没有公开,我们使用Xcode的调试器和iOS大概追溯出其实现过程。
在NSObejct类的alloc类方法上设置断点,追踪程序的执行,可以得到执行所调用的方法和函数。

可以看到alloc类方法是先调用allocWithZone:类方法和GNUstep的实现相同,然后调用class_createInstance函数,末了调用calloc来分配内存块。和前面GNUstep的实现差别不大。
那retainCount/retain/release实例方法又是怎么实现的呢?同刚才一样,先列出调用的方法和函数

可以看到每个方法都通过调用同一个__CFDoExternRefOperation函数,调用了一系列名称相似的函数。这些函数名前缀为"CF",它们包罗于Core Foundation框架源代码中。下面给出简化了CFDoExternRefOperation函数后的源代码(包罗在CFRuntime.c中)
  1. int __CFDoExternRefOperation(uintptr_t op, id obj) {
  2.     CFBasicHashRef table = 取得对象对应的散列表(obj);
  3.     int count;
  4.    
  5.     switch(op) {
  6.     case OPERATION_retainCount:
  7.       count = CFBasicHashGetCountOfKey(table, obj);
  8.       return count;
  9.     case OPERATION_retain:
  10.       CFBasicHashAddValue(table, obj);
  11.       return obj;
  12.     case OPERATION_release:
  13.       count = CFBasicHashRemoveValue(table, obj);
  14.       return 0 == count;
  15.     }
  16. }
复制代码
可以看到__CFDoExternRefOperation函数按retainCount/retain/release操纵进行分发,调用不同的函数,NSObejct类的retainCount/retain/release实例方法也许如下面代码所示:
  1. - (NSUInteger)retainCount {
  2.     return (NSUInteger)__CFDoExternRefOperation(OPERATION_retain, self);
  3. }
  4. - (id)retain {
  5.     return (id)__CFDoExternRefOperation(OPERATION_retain, self);
  6. }
  7. - (void)release {
  8.     return __CFDoExternRefOperation(OPERATION_release, self);
  9. }
复制代码
可见,苹果对内存管理的实现大概就是接纳散列表(引用计数表)来管理引用计数。

GNUstep将引用计数保存在对象占用内存块头部的变量中,而苹果的实现,则是保存在引用计数表的记载中,两者各有好处:
通过内存块头部管理引用计数有以下好处:


  • 少量代码即可完成
  • 能够统一管理引用计数用内存块与对象用内存块
通过引用计数表管理引用计数有以下好处:


  • 对象用内存块的分配无需考虑内存块头部
  • 引用计数表各记载中存有内存块地址,可以从各个记载追溯到各对象的内存块
这里第二条特性在调试时有很重要的作用,纵然出现故障导致对象占用的内存块损坏,但只要引用计数表没有被破坏,就能确认各内存块的位置

autorelease

autorelease就是自动释放,类似于C语言中局部变量的特性。autorelease会像C语言的局部变量那样来对待对象实例,当超出其作用域时,对象实例的release实例方法被调用。但是与C语言的局部变量不同的是,编程人员可以设定变量的作用域
autorelease的详细使用方法如下:

  • 天生并持有NSAutoreleasePool对象
  • 调用已分配对象的autorelease实例方法
  • 废弃NSAutoreleasePool对象
NSAutoreleasePool对象的生存周期相当于C语言变量的作用域。对于全部调用过autorelease实例方法的对象,在废弃NSAutoreleasePool对象时,都将调用release实例方法。用源代码表现如下:
  1. NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
  2. id obj = [[NSObject alloc] init];
  3. [obj autorelease];
  4. [pool drain];
复制代码
在Cocoa框架中,相当于程序主循环的NSRunLoop或者在其他程序可运行的地方,对NSAutoreleasePool对象进行天生、持有和废弃处理。因此,开辟者不肯定非得使用NSAutoreleasePool对象来进行开辟工作

只管如此,在大量产生autorelease的对象时,只要不废弃NSAutoreleasePool对象,那么天生的对象就不能被释放,因此偶然会产生内存不敷的现象。
Cocoa框架中有许多类方法用于返回autorelease的对象,比如NSMutableArray类的arrayWithCapacity类方法。
  1. id array = [NSMutableArray arrayWithCapacity:1];
  2. id array = [[[NSMutableArray alloc] initWithCapacity:1] autorelease];
复制代码
autorelease实现

首先看GNUstep的源代码:
  1. [obj autorelease]
复制代码
  1. - (id)autorelease{
  2.     [NSAutoreleasePool addObject:self];
  3. }
复制代码
autorelease实例方法的本质就是调用NSAutoreleasePool对象的addObject类方法。
下面是简化之后的NSAutoreleasePool类的源代码:
  1. + (void)addObject:(id)anObj {
  2.     NSAutoreleasePool *pool = 取得正在使用的NSAutoreleasePool对象;
  3.     if (pool != nil) {
  4.     [pool addObject:anObj];
  5.     } else {
  6.     NSLog(@"NSAutoreleasePool对象非存在状态下调用autorelease");
  7.     }
  8. }
复制代码
假如嵌套天生或持有的NSAutoreleasePool对象,理所当然会使用最内侧的对象。比如下例中pool2为正在使用的NSAutoreleasePool对象
  1. NSAutoreleasePool *pool0 = [[NSAutoreleasePool alloc] init];
  2.   NSAutoreleasePool *pool1 = [[NSAutoreleasePool alloc] init];
  3.     NSAutoreleasePool *pool2 = [[NSAutoreleasePool alloc] init];
  4.     id obj = [[NSObject alloc] init];
  5.     [obj autorelease];
  6.     [pool2 drain];
  7.   [pool1 drain];
  8. [pool0 drain];
复制代码
这里addObject实例方法是这样实现的:
  1. - (void)addObject:(id)anObj {
  2.     [array addObject:anObj];
  3. }
复制代码
实际的GNUstep实现使用的是连接列表,这同在NSMutableArray对象中追加对象参数是一样的。假如调用NSObject类的autorelease实例方法,该对象将被追加到正在使用的NSAutoreleasePool对象中的数组里。既然NSAutoreleasePool实例底层是用数组实现的,那么在使用drain实例方法时同样有处理数组的过程
  1. [pool drain];
复制代码
  1. - (void)drain {
  2.     [self dealloc];
  3. }
  4. - (void)dealloc {
  5.     [self emptyPool];
  6.     [array release];
  7. }
  8. - (void)emptyPool {
  9.     for (id obj in array) {
  10.         [obj release];
  11.     }
  12. }
复制代码
这样可以确保对数组中的全部对象都调用了release实例方法
苹果的实现

可通过objc4库的runtime/obj-arr.mm来确认苹果中autorelease的实现。
  1. class AutoreleasePoolPage {
  2.     static inline void *push() {
  3.       相当于生成或持有NSAutoreleasePool类对象;
  4.     }
  5.     static inline void *pop(void *token) {
  6.       相当于废弃NSAutoreleasePool类对象
  7.       releaseAll();
  8.     }
  9.     static inline id autorelease(id obj) {
  10.     相当于NSAutoreleasePool类的addObject类方法
  11.     AutoreleasePoolPage *autoreleasePoolPage = 取得正在使用的AutoreleasePoolPage实例;
  12.     autoreleasePoolPage->add(obj);
  13.     }
  14.     id *add (id obj) {
  15.       将对象追加到内部数组中;
  16.     }
  17.     void releaseAll() {
  18.       调用内部数组中对象的release实例方法;
  19.     }
  20. };
  21. void *objc_autoreleasePoolPush(void) {
  22.       return AutoreleasePoolPage::push();
  23. }
  24. void objc_autoreleasePoolPop(void *ctxt) {
  25.       AutoreleasePoolPage::pop(ctxt);
  26. }
  27. id *objc_autorelease(id obj) {
  28.       return AutoreleasePoolPage::autorelease(obj);
  29. }
复制代码
C++类中虽然有动态数组的实现,但其行为和GNUstep的实现完全相同。

免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
回复

使用道具 举报

0 个回复

倒序浏览

快速回复

您需要登录后才可以回帖 登录 or 立即注册

本版积分规则

干翻全岛蛙蛙

金牌会员
这个人很懒什么都没写!
快速回复 返回顶部 返回列表