iOS的内存管理,相信大家都不陌生,之前是使用的MRC,由开发人员手动来管理内存,后来使用了ARC,来由系统管理内存。本文主要讲讲Autorelease,Core Foundation对象在内存管理方面要注意的地方。

Autorelease

提到内存管理,就不得不提autorelease,虽然我们平时开发中很少会感知到它的存在。autorelease就是自动释放的意思,如果变量使用autorelease来修饰,就表明变量的释放由系统来完成。

autoreleasepool是由runloop在开启或者唤醒的时候创建的,当runloop进入睡眠或者释放掉的时候,autoreleasepool会给pool中的所有对象发送release消息。那么,由此便引申出一个问题,如果runloop不进入睡眠或者不释放(例如:主线程,或者某些常驻线程),pool里面的对象也便不会被释放,他们会堆积在内存中,但是系统会做一些优化,如下:

- (NSMutableArray*)createArrayNoAutorelease{
id arr = [NSMutableArray arrayWithCapacity:3];
return arr;
} - (NSMutableArray*)createArrayAutorelease{
return [NSMutableArray arrayWithCapacity:3];
}

上面的两个方法都是表明的要返回NSMutableArray这个对象,但是两种写法不同,系统做的处理也不相同。

我们先调用第一个方法 createArrayNoAutorelease方法,然后使用xcode的Product->Perform Action->Assemble xxx来看看,生成如下代码:

Lfunc_begin1:
....前面省略
bl _objc_retainAutoreleasedReturnValue
.loc 2 56 14
str r0, [sp, #4]
.loc 2 57 12 is_stmt 1
ldr r0, [sp, #4]
bl _objc_retain
add r1, sp, #4
movs r2, #0
.loc 2 58 1
str r0, [sp] @ 4-byte Spill
mov r0, r1
mov r1, r2
bl _objc_storeStrong
ldr r0, [sp] @ 4-byte Reload
.loc 2 58 1 is_stmt 0
add sp, #16
Ltmp3:
pop.w {r7, lr}
b.w _objc_autoreleaseReturnValue
Ltmp4:
Lfunc_end1:

其中 objc_retainAutoreleasedReturnValue和objc_autoreleaseReturnValue主要用于优化程序运行。本来应该将返回的对象注册到autoreleasepool中,但是有了这两个函数,就可以不将对象注册到autoreleasepool中,而是直接传递给调用方,这是性能调优的一个举措。

我们再来看看调用createArrayAutorelease方法,如下:

Lfunc_begin0:
... 省略
bl _objc_retainAutoreleasedReturnValue
add r1, sp, #4
movs r2, #0
.loc 2 44 8
str r0, [sp, #4]
.loc 2 49 1 is_stmt 1
mov r0, r1
mov r1, r2
bl _objc_storeStrong
add sp, #24
pop {r7, pc}
Ltmp1:
Lfunc_end0:

现在对象被注册到了autoreleasepool中。我们可以使用:

po [NSAutoreleasePool showPools]

来看看当前autoreleasepool的状况,会发现多出了一个对象,如下图:

(我只截取了一部分)

注意:对于alloc/new/copy/mutableCopy这样的方法作为返回对象,编译器会将他们优化为createArrayNoAutorelease相同的情况

关于autoreleasepool的内部结构,实现原理等,可以参看:

http://www.cocoachina.com/ios/20150610/12093.html

Core Foundation

Core Foundation对象是一组由c语言接口,可以跟Foundation框架的OC对象相互转换。

要搞清楚Core Foundation对象的内存管理,就需要搞清楚:__bridge, __bridge_retained, __bridge_transfer; CFRetain(), CFRelease() 这几个关键词的概念。

CFRetain(), CFRelease() :Core Foundation对象的内存管理方式,跟之前MRC时代的retain和release很像。

    {
CFStringRef str = CFStringCreateWithCString(kCFAllocatorDefault, "test", kCFStringEncodingUTF8);
NSLog(@"%@", str);
CFRelease(str);
}

注意:这里要调用CFRelease(str)方法,不然会有内存泄漏。

__bridge:只做类型转换,不修改对象持有者。如下:

    {
CFStringRef str = CFStringCreateWithCString(kCFAllocatorDefault, "test", kCFStringEncodingUTF8);
NSString *obj = (__bridge NSString*)str;
NSLog(@"%@", obj);
// CFRelease(str);
}

对于从CF转换为OC对象,一定要调用CFRelease(str)方法,不然会有内存泄漏,因为只是做了简单的指针地址变换,str仍然没有释放。

来看一个野指针的例子:

    CFMutableArrayRef cfObject = NULL;
{
id obj = [[NSMutableArray alloc] init];
cfObject = (__bridge CFMutableArrayRef)obj;
CFShow(cfObject);
}
CFRelease(cfObject);

因为__bridge不持有obj对象,所以当大括号结束以后,obj被释放,cfObject就成为了野指针,在调用CFRelease方法时就会引发程序崩溃。

__bridge_retained:用于将OC对象转换为CF对象,持有者也发生改变,需要调用CFRelease方法。

__bridge_transfer:将CF对象转换为OC对象,同时将对象持有权交给ARC,不需要调用CFRelease方法,如下:

    {
CFStringRef str = CFStringCreateWithCString(kCFAllocatorDefault, "test", kCFStringEncodingUTF8);
NSString *obj = (__bridge_transfer NSString*)str;
NSLog(@"%@", obj);
// CFRelease(str);
}

所以在使用CF对象时,要特别注意内存问题。

参考文章:

https://yq.aliyun.com/articles/58964

https://juejin.im/entry/579bfdfe5bbb500064d18aca

iOS之内存管理(ARC)的更多相关文章

  1. 理解 iOS 的内存管理

    远古时代的故事 那些经历过手工管理内存(MRC)时代的人们,一定对 iOS 开发中的内存管理记忆犹新.那个时候大约是 2010 年,国内 iOS 开发刚刚兴起,tinyfool 大叔的大名已经如雷贯耳 ...

  2. 说说iOS与内存管理(上)

    http://www.cocoachina.com/ios/20150625/12234.html 说起内存管理,看似老生常谈,而真正掌握内存管理的核心其实并不简单.ARC/MRR以及“谁分配谁就负责 ...

  3. iOS的内存管理和引用计数规则、Block的用法以及三种形式(stack、malloc、global)

    学习内容 iOS的内存管理和引用计数规则 内存管理的思考方式 自己生成的对象自己持有 非自己生成的对象自己也能持有 自己持有的对象不需要时释放 非自己持有的对象不能释放 ARC有效时,id类型和对象类 ...

  4. iOS内存管理 ARC与MRC

    想驾驭一门语言,首先要掌握它的内存管理特性.iOS开发经历了MRC到ARC的过程,下面就记录一下本人对iOS内存管理方面的一些理解. 说到iOS开发,肯定离不开objective-c语言(以下简称OC ...

  5. IOS基础之 (十一) 内存管理 ARC

    一 内存管理 1. set 方法内存管理的相关参数 retain: release旧值,retain新值(值适用于OC对象) assign:直接赋值(set方法默认,适用于非OC对象类型,即基本数据类 ...

  6. OC内存管理(ARC)

    1.什么是ARC Automatic Reference Counting,自动引用计数,即ARC,可以说是WWDC2011和iOS5所引入 的最大的变革和最激动人心的变化.ARC是新的LLVM 3. ...

  7. iOS - OC 内存管理

    1.OC 基本内存管理模型 1.1 自动垃圾收集 在 OC 2.0 中,有一种称为垃圾收集的内存管理形式.通过垃圾收集,系统能够自动监测对象是否拥有其他的对象,当程序执行需要空间的时候,不再被引用的对 ...

  8. 总结 IOS 7 内存管理

    [iOS7的一些总结].iOS中的内存管理 我们知道,为了更加方便地处理内存管理问题,将开发人员从繁琐的内存的分配和释放工作中解放出来而专注于产品和逻辑,iOS提供了一种有效的方法, 即自动引用计数A ...

  9. iOS的内存管理

    在Objective-C 这种面向对象的语言里,内存管理是个重要的概念.要想用一门语言写出内存使用效率高而且又没有bug的代码,就得掌握其内存管理模型的种种细节. 一旦理解了这些规则,你就会发现,其实 ...

随机推荐

  1. uboot 添加hello命令

    平台:MPC8315(POWERPC) 1.在/common/ 目录下创建自己的文件,最好前缀为cmd_. cmd_hello.c ********************************** ...

  2. linux下编译sphinx拓展

    编译libsphinxclient sphinx 源码包里的api文件夹下的libsphinxclient cd /root/api/libsphinxclient/ ./configure make ...

  3. TI AM335X处理器介绍

    AM335X是美国TI(德州仪器)公司基于 ARM Cortex-A8内核的AM335X微处理器,在图像.图形处理.外设方面进行了增强,并全面支持诸如 EtherCAT 和 PROFIBUS等工业接口 ...

  4. C#之AES256位加密解密

    密码学中的高级加密标准(Advanced Encryption Standard,AES),又称Rijndael加密法,是美国联邦政府采用的一种区块加密标准.这个标准用来替代原先的DES,已经被多方分 ...

  5. (二十二)java小练习三

    练习七:计算1-100的累加和 package demo; /**  * 计算1-100的累加和  * @author tuzongxun  */ public class Test16 {   pu ...

  6. dojo中获取表格中某一行的某个值

    dojo中经常出现对表格中的某行进行操作,如单击某行修改.删除等.那怎样获取某行的唯一标示呢? 如查询表格中的某列有个userId,并且这个是唯一的,那么可以通过它来访问这一列 具体操作代码如下: v ...

  7. mfc100u.dll下载和使用方法

      当运行软件或游戏时,系统提示"丢失mfc100u.dll"."没有找到mfc100u.dll"等类似错误信息. 请下载本站提供的dll文件,使用它可以帮助用 ...

  8. 【BZOJ1146】网络管理(整体二分)

    [BZOJ1146]网络管理(整体二分) 题面 良心洛谷,有BZOJ权限题 题解 要看树套树的戳这里 毕竟是:智商不够数据结构来补 所以, 我们来当一回智商够的选手 听说主席树的题目大部分都可以整体二 ...

  9. Bzoj1899: [Zjoi2004]Lunch 午餐

    题面 传送门 Sol 首先显然吃饭久的要排在前面 之后再来分配队伍,设\(f[i][j]\)表示到第\(i\)个人,\(A\)队伍要等\(j\)的最小吃完饭时间 那么就是一个简单的背包吧... # i ...

  10. PHP接收android传过来的图片

    在android手机app上,上传图片,PHP服务端接收图片需要一下步骤 1 需要android 开发人员 把图片转成base64格式 2 PHP端接收转化后的base64字符串,然后进行转译 fun ...