前言:本文旨在介绍iOS性能优化中有关内存泄漏的介绍和检测。
一、什么是内存泄漏?
内存泄漏是指申请的内存空间使用完成之后没有及时的回收,导致页面不释放,从而加剧了内存的暴增。
二、常见的内存泄漏
1、Block的循环应用
一个对象持有Block,同时Block内部又持有这个对象,会导致循环应用,相互不释放。外部可以使用__weak来进行解决。
2、代理delegate用strong修饰
代理delegate用strong修饰同样会造成内存泄漏。可以使用weak修饰delegate来解决这个问题。
3、NSTimer定时器的使用
3.1、创建定时器使用了带有target的方法
+ (NSTimer *)timerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo;
+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)ti target:(id)aTarget selector:(SEL)aSelector userInfo:(nullable id)userInfo repeats:(BOOL)yesOrNo;
上面的方法创建的NSTimer,会使其不能释放。可以使用以下两种方式解决:
- 方式一:设置target时,使用NSProxy来解决,详情可参考YYKit中的使用创建方式:
@interface YYWeakProxy : NSProxy
@property (nullable, nonatomic, weak, readonly) id target;
- (instancetype)initWithTarget:(id)target;
+ (instancetype)proxyWithTarget:(id)target;
@end
self.timer = [NSTimer timerWithTimeInterval:1 target:[YYWeakProxy proxyWithTarget:self] selector:@selector(setupCountAction) userInfo:nil repeats:YES];
备注:NSProxy也是一个基类,和NSObject同级别的基类。主要用来做消息转发。
- 方式二:直接使用带有block的创建方式:
+ (NSTimer *)timerWithTimeInterval:(NSTimeInterval)interval repeats:(BOOL)repeats block:(void (^)(NSTimer *timer))block API_AVAILABLE(macosx(10.12), ios(10.0), watchos(3.0), tvos(10.0));
+ (NSTimer *)scheduledTimerWithTimeInterval:(NSTimeInterval)interval repeats:(BOOL)repeats block:(void (^)(NSTimer *timer))block API_AVAILABLE(macosx(10.12), ios(10.0), watchos(3.0), tvos(10.0));
- (instancetype)initWithFireDate:(NSDate *)date interval:(NSTimeInterval)interval repeats:(BOOL)repeats block:(void (^)(NSTimer *timer))block API_AVAILABLE(macosx(10.12), ios(10.0), watchos(3.0), tvos(10.0));
不过以上方法都是iOS10才出现的API,使用时需要注意。
3.2、创建的定时器在页面消失后没有及时的对Timer进行销毁
一般情况下,我们会在定时器结束后去销毁定时器。但出现以下情况则会出现无法销毁定时器的问题:
- 情况一:当前页面定时器还未结束,而页面已经返回上一页;
- 情况一:当前页面定时器还未结束,而此时跳转到下一页,由于一些操作最后并没有回到这个页面,而此时的定时器还在执行。
解决办法:可以在页面的 viewDidAppear 和 viewDidDisappear 的两个方法中根据具体的情况去创建和销毁定时器。这样在页面消失的时候就可以及时的销毁定时器了。
三、内存泄露检测工具:
1、腾讯 MLeaksFinder
1、优点和缺点
- 优点:对项目无侵入。将MLeaksFinder 引入到项目中,操作项目页面,有内存泄漏问题,则会弹窗提示。
- 缺点:目前只能对相关Controller、UIView作出检测,对于其他类则无法作出检测。
2、检测原理:
首先通过分类的方式给基类NSObject添加一个 willDealloc 的方法,该方法中会生成一个弱指针指向weakSelf,并在2秒之后 weakSelf 会调用弹窗提示方法 assertNotDealloc。然后通过方法交换的方式,当VC 执行pop 或 dismiss 后,在该方法中去执行willDealloc方法。若pop 或 dismiss后对象被销毁,dealloc执行后weakSelf会指向nil,就不会调用assertNotDealloc方法。反之,就会调用弹窗,给出提示。
2、腾讯 OOMDetector
具体检测内容:
- OOM监控:监控OOM,Dump引起爆内存的堆栈
- 大内存分配监控:监控单次大块内存分配,提供分配堆栈信息
- 内存泄漏检测:可检测OC对象、Malloc堆内存泄漏,提供泄漏堆栈信息
3、Facebook的 FBRetainCycleDetector
检测原理:
对所有检测到的强引用变量,利用DFS(深度优先搜索),采用stack栈的形式,将遍历到的对象依次入栈,如果某次入栈的对象已经存在stack中,说明该对象在stack中的位置直到当前为止,存在引用环。