收藏的简书iOS精学选辑iOS知识积累

iOS 使用Leak进行内存泄漏的分析

2017-02-22  本文已影响2004人  晴天WJ

第一次使用Leak对项目进行内存泄漏的检测,也是查阅了许多资料,算是可以做到发现问题并解决问题。

我的项目泄漏情况是这样的:

图1

满屏的红叉让人心慌,那么如何找到泄露的位置呢?

首先要选中满是红叉的那一行,然后在这里

图2

选中Call Tree,

在右边

图3

选中Invert Call Tree 和 Hide System Libraries 两项,Invert Call Tree 的意思是翻转调用树,意思就是我们在调用函数时,是一层一层的,调用外层函数会一直进入内层,直到最后一层,有点递归的意思,当选中 Invert Call Tree 选项时,会直接显示内层函数,方便我们去寻找,否则会直接显示最外层的函数,我们需要将其一层一层展开,比较费劲,不直观。

而Hide System Libraries 的意思很明显了。就是隐藏系统类库,避免一些莫名其妙的,我们无法改动的信息迷惑我们。

那么接下来我们将看到泄漏列表:

图4

看到这里我震惊了,强大的AFNetworking也会存在泄露?

让我们再看看具体是哪里除了问题,双击那一行我们就可以进入到具体泄漏的那个函数

图5

可以看到每一行泄漏的byte大小都标了出来,其中蓝色的为最大。

这里就要思考一个问题了,AFNetworking内部的这些代码我们怎么改呢?

不必担心,我们发现问题出在这个方法

AFHTTPSessionManager *manager = [AFHTTPSessionManager manager];

一直以来,我都以为这是一个manager是单例,带你进去一看其实不然,

图6

每次调用时都会创建一个新的对象,那么问题又来了,创建就创建呗,难道执行之后,还会不释放?

查阅资料后果真如此,我们使用的 AFHTTPSessionManager 继承自 AFURLSessionManager ,

创建对象时会调用傅父类的方法。

图7

点进去看,发现其强引用一个NSURLSession对象

图8

并且将自己设置为了NSURLSession对象的代理


图9

NSURLSession又是强引用代理


图10

这样便造成了循环引用,彼此谁也释放不了。

那么如何解决呢?

这里提供两种策略

(一)NSURLSession提供两个方法:

图11

这个方法会立刻取消当前任务,session对象被释放。那么循环引用不复存在。

图12

而finishTasksAndInvalidate方法,则会等待任务完成时将session释放,消除了循环引用。

我们可以在success 和 failure block中调用这两个方法,个人推荐使用finishTasksAndInvalidate。

(二)像AFNetworking 3.0 提供的DEMO中,是这么用的:

创建一个继承自NSHTTPSessionManager的类,实现单例方法。

图13

然后在自己封装的网络层中修改。

上一篇下一篇

猜你喜欢

热点阅读