iOS之内存管理(待续)
2017-10-27 本文已影响13人
SK丿希望
内存管理是什么?
- 软件运行时对内存资源进行合理分配和使用的技术。其最主要的目的是高效,快速的分配资源,并且在适当的时候释放和回收内存资源。
- 其本质是管理对象的引用计数,对象的引用计数就是标记当前有多少个其他对象使用(拥有)这个对象。分为自动内存管理
引用计数
引用计数(Reference Count)是一个简单而有效的管理对象生命周期的方式。不管是Objective-C、还是Swift,其内存管理方式都是基于引用计数的。
内存管理方式
MRC(手动管理iOS5之前)
凡是使用alloc、init、copy、nsmutablecopy、retain进行创建对象的都要使用release或者autorelease进行释放;
规则:
- 当使用alloc/new/copy/retain,创建或引用一个对象时会使对象的引用计数+1。
- 当不使用对象,会向对象发送release或autorelease消息会使对象的引用计数-1。
- 当一个对象的引用计数为0时,那么它将被销毁,其占用的内存被系统回收,OC会自动向其发送一条dealloc消息(自动调用dealloc方法)
- 我们此时可以重写dealloc方法,在这里释放相关资源。一旦某个对象的引用计数为0,那么它就会被系统释放掉。只要对象的引用计数不为0,它就在内存中永远不会被释放, 这样就会造成内存泄露。
ARC(自动管理iOS5之后)
是编译时的特性,不需要我们手动retain或者release,系统会在适当的时候为我们添加相应的代码。但是自动管理内存需要我们注意循环引用以防止内存泄漏。
在ARC内存管理模式下,其属性的标识符有assign,strong,weak,copy;
- assign在arc中修饰的是基本数据类型,mrc中修饰的是delegate防止循环引用造成内存泄漏。
- strong相当于mrc中的retain,使得对象被引用时引用计数加1.
- weak的作用相当于assign,弱引用使用的对象。在arc中delegate的修饰词是weak。作用防止循环引用造成内存泄漏。
- copy修饰的类型一般是NSString,NSArray,NSDictionary。建立一个索引计数为1的对象,然后释放旧对象。
- assign和weak的区别:
首先assign修饰的是基本数据类型,简单赋值不改变引用计数,weak只可以修饰oc中的对象。其次arc环境下weak修饰的对象被释放后指向对象的指针会被自动置为nil,而assign修饰的变量可能不会被置为nil,造成野指针会导致程序crash。
iOS内存段介绍:
内存管理的好处?
iOS内存管理机制的好处就是为了让开发人员方便的管理内存,减少程序中的内存泄漏,在内存管理难度与性能之间找一个最佳的平衡点。
block的内存管理
* 循环引用
block一般用copy修饰,当block又引用了对象的其他成员变量时,就会对这个变量本身产生强引用,那么变量本身和它自己的block就形成了循环引用。
__weak typeof (self) weakSelf = self;
ARC下要生成一个对自身的弱引用,表示block别再对self对象retain了,避免循环引用
* block内部变量
1.block不能改变局部变量,要改变时需加__block修饰
2.block可改变全局变量
3.static变量也可在block中修改
内存问题的分析解决
僵尸对象和野指针
僵尸对象:内存被回收的对象
野指针:指向僵尸对象的指针,向野指针发送消息对导致崩溃EXC_BAD_ACCESS
1 在product-scheme-edit scheme-diagnostics中将enable zombie objects勾选上,下次再出现这样的错误就可以准确定位了。
2 在Xcode-open developer tool-Instruments打开工具集,选择Zombies工具可以对已安装的应用进行僵尸对象检测。
循环引用
1)在product-Analyze中使用静态分析来检测代码中可能存在循环引用的问题。
2)在Xcode-open developer tool-Instruments打开工具集,选择Leaks工具可以对已安装的应用进行内存泄漏检测,此工具能检测静态分析不会提示,但是到运行时才会出现的内存泄漏问题。
循环中对象占用内存大
循环内产生大量的临时对象,直至循环结束才释放,可能导致内存泄漏。
解决方法:在循环中创建自己的autoReleasePool,及时释放占用内存大的临时变量,减少内存占用峰值。
内存泄漏
通过Analyze来进行静态代码检查,以发现在语法上显而易见的内存泄露问题
内存泄露是运行时的问题,这时可用Instruments中的Allocation和Leaks来不断重复操作App,发现和定位内存泄露点
常用的内存优化方法
图片读取:
imageNamed//使用缓存,适用于频繁使用的小图片
imageWithContentOfFile//不使用缓存,适用于大图的读取
文档读取
+ (nullable instancetype)dataWithContentsOfFile:(NSString *)path;//
+ (nullable instancetype)dataWithContentsOfFile:(NSString *)path options:(NSDataReadingOptions)readOptionsMask error:(NSError **)errorPtr;//映射文件到虚拟内存,大文件时采用
内存警告处理
当内存不足时,就会收到内存警告。在iOS中提供了3中内存警告通知方式:
1.在应用程序委托中实现applicationDidReceiveMemoryWarning:方法: 应用程序委托对象中接收内存警告消息
2.在UIViewController子类中实现didReceiveMemoryWarning方法:视图控制器中接收内存警告消息
3.注册UIApplicationDidReceiveMemoryWarningNotification通知:其它类中使用通知接收内存警告消息
收到内存警告,一般有以下几种处理方法:
1.尽可能多的释放资源,尤其是图片等占用内存较多的
2.释放掉单例对象
3.iOS6之后对于隐藏的viewController直接设置self.view=nil