内存管理(二) - MRC关键字解读
本篇主要学习以下几个知识点
- alloc/reatin/release/dealloc 理解
- autorelease 理解
- autorelease GUN 实现
- autorelease 苹果 实现
原文地址
alloc/reatin/release/dealloc 实现
我们来看看 GUNstep 源代码中 NSObject 类的的 alloc 类方法。
id obj = [NSObject alloc];
上述调用 NSObject 类的 alloc 类方法在 NSObjecr.m 源代码中的实现如下。
+ (id) alloc
{return [self allocWithZone: NSDefaultMallocZone()];
}
+ (id) allocWithZone:(NSZone *)z
{return NSAllocateObject(self,0,z);
}
通过 allocWithZone: 类方法调用 N\NSAllocateObject 函数分配对象。下面我们查看 NSAllocateObject 函数.
struct obj_layout {NSUInterger retained;
}
inline id NSAllocateObject (Class aClass, NSUInterger extraBytes, NSZone *zone) {int size = 计算容纳对象所需内存大小;id new = NSZoneMalloc(zone, size);memset(new, 0, size);new = (id)&((struct obj_layout *) new)[1];
}
NSAllocateObject 函数通过调用 NSZoneMalloc 函数来分配存放对象所需的内存空间,之后将改内存空间置0,最后返回作为对象而使用的指针。
以下是去掉 NSZone 后简化了源代码:
struct obj_layout {NSUInterger retained;
}
+ (id) alloc {int size = sizeof(struct obj_layout) + 对象大小;struct obj_layout *p = (struct obj_layout *)calloc(1, size);return (id)(p+1);
}
alloc 类方法用 struct obj_layout 中的 retained 整数来保存引用计数,并将其写入对象内存头部,该对象内存全部置0后返回。
以下用图来展示有关 GUNstep 的实现,alloc类方法返回对象,如图
对象的引用计数可通过 retainCount 实例方法来取得。
id obj = [NSObject alloc];
NSLog(@"retainCount=%d", [obj retainCount]); // => 1
执行alloc后对象的 retainCount 是 “1”。下面通过 GUNstep 的源代码来确认。
- (NSUInteger)retainCount {return NSExtraRefCount(self) + 1;
}
inline NSUInterger NSExtraRefCount(id anObject) {return ((struct obj_layout *) anObject)[-1].retained;
}
由对象寻址找到对象内存头部,从而访问其中的retained变量。如图:
autorelease
autorelease 就是自动释放。这看上去很像ARC,但实际上它更类似于C语言中的自动变量(局部变量)的特性。
{int a;
}
/*因为超出变量作用域,自动变量 "int a" 被废弃,不可再访问
*/
autorelease 会像 C 语言的自动变量那样来对待对象实例。当超出其作用域(相当于变量作用域)时,对象实例的 release 实例方法被调用。另外,同 C 语言的自动变量不同的是,编程人员可以设定变量的作用域。
autorelease 的具体使用方法如下:
- 生成并持有 NSAutoreleasePool 对象
- 调用已分配对象的 autorelease 实例方法
- 废弃 NSAutoreleasePood 对象
NSAutoreleasePool 对象的声明周期相当于 C 语言变量的作用域。对于所有调用过 autorelease 实例方法的对象,在废弃 NSAutoreleasePool 对象时,都将调用 release 实例方法。如上图。
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];id obj = [[NSObject alloc] init];[obj autorelease];[pool drain];
“[pool drain]” 等同于 “[obj release]”。
在 Cocoa 框架中,相当于程序主循环的 NSRunLoop 或者在其他程序可运行的地方,对 NSAutoreleasePool 对象进行生成、持有和废弃处理。因此,应用程序开发者不一定非得使用 NSAutoreleasePool 对象来进行开发工作。
意思就是我们不一定需要去管理 NSAutoreleasePool。
尽管如此,但在大量产生 autorelease 的对象时,只要不废弃 NSAutoreleasePool 对象,那么生成的对象就不能被释放,因此有时会产生内存不足的现象。
典型的例子:读入大量图像的同时改变尺寸。图像文件读入到 NSData 对象,并从中生成 UIImage 对象,改变该对象尺寸后生成新的 UIImage 对象。这种情况下,就会大量生成 autorelease 对象。
for (int i = 0; i < 图像数; ++i){/*读入图像大量产生 autorelease 对象由于没有废弃 NSAutoreleasePool 对象最终导致内存不足*/
}
在此情况下,有必要在适当的地方生成,持有或者废弃 NSAutoreleasePool 对象。
for (int i = 0; i < 图像数; ++i){NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];/*读入图像大量产生 autorelease 对象*/[pool drain];
}
另外,Cocoa 框架中也有很多类方法用于返回 autorelease 的对象。比如 NSMutableArray 类的 arrayWithCapacity 类方法。
id array = [NSMutableArray arrayWithCapacity:1];
此源代码等同于以下源代码
id array = [[[NSMutableArray alloc] initWithCapacity:1] autorelease];
autorelease 实现 - GUNstep是如何实现的
[obj autorelease];
此源代码调用 NSObject 类的 autorelease 实例方法。
GUNstep实现:
- (id) autorelease {[NSAutoreleasePool addObject:self];
}
autorelease 实例方法的本质就是调用 NSAutoreleasePool 对象的 addObject 类方法。
+ (void) addObject:(id)anObj {NSAutoreleasePool *pool = 取得正在使用的 NSAutoreleasePool 对象;if (pool != nil) {[pool addObject: anObjc];}else{NSLog(@"NSAutoreleasePool 对象非存在状态下调用 autorelease")}
}
addObject 类方法调用正在使用的 NSAutoreleasePool 对象的 addObject 实例方法。
如果嵌套或者持有 NSAutoreleasePool 对象,理所当然会使用最内侧的对象。
- (void) drain {[self dealloc];
}
- (void) dellloc {[self emptyPool];[array release];
}
- (void) emptyPool {for (id obj in array){[obj release];}
}
虽然调用了好几个方法,但是可以确定对于数组中的所以对象都调用了 release 实例方法。
苹果 autorelease 的实现
objc4 库的 runtime/objc-arr.mm 来确认苹果中 autorelease 的实现
Class AutoreleasePoolPage
{static inline void *push (){相当于生成或者持有 NSAutoreleasePool 对象}static inline void *pop (void *token){releaseAll();}static inline id autorelease (id obj){相当于 NSAutoreleasePool 类的 addObject 类方法NSAutoreleasePoolPage *autoreleasePoolPage = 取得正在使用的实例;autoreleasePoolPage ->add(obj);}id *add (id obj) {将对象追加到内部数组中}void releaseAll(){调用内部数组中对象的release实例方法}
}
void *objc_autoreleasePoolPush(void){return AutoreleasePoolPage::push();
}
void *objc_autoreleasePoolPop(void *ctxt){return AutoreleasePoolPage::pop(ctxt);
}
id *obj_autorelease(id obj) {return AutoreleasePoolPage:autorelease(obj);
}
C++类中虽然有动态数组的实现,但其行为和GUNstep的实现完全相同。
我们使用调试器来观察一下NSAutoreleasePool类方法和 autorelease 方法的运行过程。如下所示,这些方法调用了关联于 objc4 库的 autorelease 实现的函数。
NSAutoreleasePool *pool = [[NSAutoreleasePool alloc] init];
/*= objc_autoreleasePoolPush()*/id obj = [[NSObject alloc] init];[obj autorelease];
/*= obj_autorelease(obj)*/[pool drain];
/*= objc_autoreleasePoolPop(pool)*/
另外,可通过 NSAutoreleasePool 类中的调试用非公开类方法 showPools 来确认已被 autorelease 的对象情况。
[NSAutoreleasePool showPools];
NSAutoreleasePool 类的 showPools 类方法只能在 iOS 中使用,作为替代,在现在的运行时系统中我们使用调试用非公开函数 _objc_autoreleasePoolPrint()
。
/* 函数声明 */
extend void _objc_autoreleasePoolPrint();/* autorelesepool 调试用输出开始*/
_objc_autoreleasePoolPrint();
如果运行此函数,就能像下面这样在控制台中来确认 AutoreleasePoolPage 类的情况。
NSAutoreleasePool 调用 autorelease 会如何?发生异常
通常在使用 Obejctive-C,也就是 Foundation 框架时,无论调用哪一个对象的 autorelease 实例方法,实际上调用都是 NSObject 类的 autorelease 实例方法。但是对于 NSAutoreleasePool 类,autorelease 实例方法已被该类重载,因此运行时就会报错。
内存管理(二) - MRC关键字解读相关推荐
- iOS 开发:彻底理解 iOS 内存管理(MRC 篇)
本文首发于我的个人博客:「程序员充电站」 文章链接:「传送门」 本文更新时间:2021年08月17日17:11:59 本文是 「iOS 开发:彻底理解 iOS 内存管理」系列的「MRC 篇」. 用来对 ...
- OC 知识:彻底理解 iOS 内存管理(MRC、ARC)
1. 什么是内存管理 程序在运行的过程中通常通过以下行为,来增加程序的的内存占用 创建一个OC对象 定义一个变量 调用一个函数或者方法 而一个移动设备的内存是有限的,每个软件所能占用的内存也是有限的 ...
- 操作系统概念学习笔记 16 内存管理(二) 段页
操作系统概念学习笔记 16 内存管理 (二) 分页(paging) 分页(paging)内存管理方案允许进程的物理地址空间可以使非连续的.分页避免了将不同大小的内存块匹配到交换空间上(前面叙述的内存管 ...
- Linux任督二脉之内存管理(二) PPT
五节课的第二节课-内存的动态申请和释放 * slab.kmalloc/kfree./proc/slabinfo和slabtop * 用户空间malloc/free与内核之间的关系 * mallopt ...
- Linux内存管理二(页表)
1.综述 用来将虚拟地址空间映射到物理地址空间的数据结构称为页表,即页表用于建立用户进程的虚拟地址空间和系统物理内存(内存.页帧)之间的关联 实现两个地址空间的关联最容易的方法是使用数组,对虚拟地址空 ...
- 高端内存映射之kmap持久内核映射--Linux内存管理(二十)
日期 内核版本 架构 作者 GitHub CSDN 2016-09-29 Linux-4.7 X86 & arm gatieme LinuxDeviceDrivers Linux内存管理 在内 ...
- BOOST内存管理(二) --- boost::pool
Boost库的pool提供了一个内存池分配器,用于管理在一个独立的.大的分配空间里的动态内存分配.Boost库的pool主要适用于快速分配同样大小的内存块,尤其是反复分配和释放同样大小的内存块的情况. ...
- iphone内存管理(二)
(2)尽量避免使用autorelease 虽然autorelease非常简单有用,但是在iphone上一定要谨慎使用,毕竟iphone内存相当有限.autorelease可能会导致直接的隐型内存泄露. ...
- matlab内存管理(二)
转自:http://hi.baidu.com/bi%CB%AB%C9%FA%BB%A8/blog/item/5ab86c38ac2f45e715cecbab.html 1,确保内存的连续性 Matla ...
最新文章
- GoCart 分类和产品 测试一
- lnmp环境切换php版本,并安装相应redis扩展
- SharePoint 集成OWA概述
- 【鬼网络】之部署YUM仓库大全
- sqlalchemy limit_SQLAlchemy 数据库方言(dialect)是什么?
- Spring MVC常用注解,你会几个?
- 自制简单的小型Linux,自制小型Linux系统
- 作者:向阳,男,同济大学教授、博士生导师。
- linux实现单机qq_Linux后台服务器开发——Linux下进程间通信的方式有哪些?
- python 遍历文件夹 文件
- Jmeter 常用断言使用
- 扫描工具——Nmap用法详解
- 测试驱动开发(TDD)实践与技巧
- 网上商城系统支付方式如何配置?支付方式有哪些
- 比较程序查询方式、程序中断方式、DMA方式对CPU工作效率的影响
- 生物计算机是未来型计算机吗,生物计算机特点及未来发展
- Webstorm基本配置
- 值得品读的感悟人生的经典句子 - 格言网(转载)
- 基于PyTorch的生成对抗网络入门(3)——利用PyTorch搭建生成对抗网络(GAN)生成彩色图像超详解
- 【转自JULY大佬】程序员面试、算法研究、编程艺术、红黑树、机器学习5大系列集锦
热门文章
- 将 zlib 引入到C++项目中 历程
- Word中如何快速删除页眉下的横线?教你一招,轻松解决!
- QObject::killTimer: Timers cannot be stopped from another thread
- Huffman 编码的实现(C语言)
- 拼多多存在多种重大风险
- python writeline_Python文件写入函数 write()和writelines()
- Pytorch深度学习(一):前馈神经网络(FNN)
- 获取农历节日的公共方法
- 取消GitHub邮箱订阅
- 跨界融合,共创智能汽车研发新生态(技术大会诚邀您的莅临)