编码篇-ARC下的内存泄漏


</br>

前言

内存泄露是一个相对挺严重的问题,可是它的存在未引起足够的重视,如果程序运行时一直分配内存而不及时释放无用的内存,程序占用的内存越来越大,直到把系统分配给该APP的内存消耗殚尽,程序因无内存可用导致崩溃,这样的情况我们称之为内存泄漏。如果某个对象没有始终在内存中,并且依然会做一些事的时候,这样的的Bug是非常严重而且难以排查的。

内存泄漏可能引起的问题:

  • 内存消耗殆尽的时候,程序会因没有内存被杀死,即crash。
  • 当内存快要用完的时候,会非常的卡顿
  • 如果是ViewController没有释放掉,引起的内存泄露,还会引起其他严重的问题,尤其是和通知相关的。没有被释放掉的ViewController还能接收通知,还会执行相关的动作,所以会引起各种各样的异常情况的发生。

那么ARC下内存泄漏的场景有哪些呢

值得注意的是:ARC是编译器(时)特性,而不是运行时特性,更不是垃圾回收器(GC)。
ARC这是一种编译期的内存管理方式,在编译期间,编译器会判断对象的使用情况,并在合适的位置加上retain和release,使得对象的内存被合理的管理。所以,从本质上说ARC和MRC在本质上是一样的,都是通过引用计数的内存管理方式。

  • CF类型内存

ARC 可以帮忙管理 Objective-C 对象, 但是不支持 Core Foundation 对象的管理,所以转换后要注意一个问题:谁来释放使用后的对象。
注意以creat,copy作为关键字的函数都是需要释放内存的,注意配对使用。比如:CGColorCreate<-->CGColorRelease

那Objective-C 和 Core Foundation 对象相互转换时就可能出现内存泄漏的问题,可参考这篇文章处理。

  • MRC内存使用

    这部分不做详细介绍,也是注意配对使用,需要说明的是,如果代码中有部分文件是MRC的,在已有文件中加代码的时候注意一下,不能都按照ARC的方式处理。

  • 循环引用

    • block引起的循环引用。
      某个类将block作为自己的属性变量,然后该类在block的方法体里面又使用了该类本身;相互持有,导致都释放不了。
      下面这样的方式就可以解决block引起的循环引用:
      __weaktypeof(self) weakSelf =self;
      block内的self,换成weakSelf就行了。
      block不是self的属性或者变量时,在block内使用self不会循环引用;
      像这样的方法中调用self,不会引起,但是属性的形式中调用self就会以

        [self.myTest doSomeTest:^(NSInteger cellIndex) {
            self.allInter = cellIndex;
        }];
      
  • 引用大循环
    ​就像前面说的,引用循环可能是一个大循环。我遇到过一种情况,就是给UITableViewCell设置block属性响应事件,在block中强引用了self,
    导致self->tableView->cell->self形成循环。
    有时候随着代码量的增大,逻辑的负责,很容易形成一个很大的循环引用,最后造成内存泄漏。

  • ** NSTimer的使用**

    NSTimer,NSTimer会对它的target持有强引用,如果NSTimer不释放掉,就会一直持有它的target的强引用,如果这个NSTimer在被target强引用,会一直都释放不掉,造成内存泄露。

    下面的代码在书写的时候Xcode是不会报任何错误和警告的。但是实际上已经形成了循环引用。造成了内存泄漏。

    @property (nonatomic, strong) NSTimer *timer;
    @property(copy,nonatomic)NSString *name;
     self.timer = [NSTimer scheduledTimerWithTimeInterval:0.1
                                                target:self
                                              selector:@selector(handleTimer)
                                              userInfo:nil
                                               repeats:YES];
    
      - (void)handleTimer
      {
           self.name = @"123";
     }
    
  • 单例也会造成内存泄漏

    如果一个单例持有一个block,block内又使用了当前这个ViewController类,会引起循环引用。所以单例持有的代码块中要用弱引用,原因是:单例不会被释放掉,它会一直持有block,导致该block所在的ViewController释放不掉。

  • performSelector的内存问题

    • performSelector 的动态绑定

      SEL selector;
        if (/* some condition */) {
        selector = @selector(newObject);
       } else if (/* some other condition */) {
        selector = @selector(copy);
       } else {
         selector = @selector(someProperty);
      }
      id ret = [object performSelector:selector];
      

    这段代码就相当于在动态之上再动态绑定。在 ARC 下编译这段代码,编译器会发出警告

          warning: performSelector may cause a leak because its selector is unknow [-Warc-performSelector-leak]
    

    正是由于动态,编译器不知道即将调用的 selector 是什么,不了解方法签名和返回值,甚至是否有返回值都不懂,所以编译器无法用 ARC 的内存管理规则来判断返回值是否应该释放。因此,ARC 采用了比较谨慎的做法,不添加释放操作,即在方法返回对象的引用计数可能不会减少,从而可能导致内存泄露。

    以本段代码为例,前两种情况(newObject, copy)都需要再次释放,而第三种情况不需要。这种泄露隐藏得如此之深,以至于使用 static analyzer 都很难检测到。如果把代码的最后一行改成

        [object performSelector:selector];
    

    不创建一个返回值变量测试分析,简直难以想象这里居然会出现内存问题。所以如果你使用的 selector 有返回值,一定要处理掉 手动释放(置为 nil)。

    • performSelector afterDelay 延时操作
      关于内存管理的执行原理是这样的执行
      [self performSelector:@selector(method1:) withObject:self.tableLayer afterDelay:3];
      的时候,系统会将tableLayer的引用计数加1,执行完这个方法时,还会将tableLayer的引用计数减1,有时切换场景时延时函数已经被调用但还没有执行,这时tableLayer的引用计数并没有减少到0,也就导致了切换场景dealloc方法没有被调用,出现了内存泄露。

      解决办法就是取消那些还没有来得及执行的延时函数,代码:

      [NSObject cancelPreviousPerformRequestsWithTarget:self]
      

      当然你也可以一个一个得这样用:

        [NSObject cancelPreviousPerformRequestsWithTarget:self selector:@selector(method1:) object:nil]
      

      加上了这个以后,切换场景后会顺利地执行了dealloc方法,至此内存泄漏问题解决。

  • 代理未清空引起野指针

    查看iOS的一些API,发现delegate都是assign的,这样就会引起野指针的问题,可能会引起一些莫名其妙的crash。那么这是怎么引起的,当一个对象被回收时,对应的delegate实体也就被回收,但是delegate的指针确没有被nil,从而就变成了游荡的野指针了。所以在delloc方法中要将对应的assign代理设置为nil,如:

    - (void)viewDidDisappear:(BOOL)animate
    {
      self.myTableView.delegate = nil;
      self.myTableView.dataSource = nil;
      通知注销掉
      kvo remove掉~~
    }
    

那是不是所有的delegate都要这样做呢?一般自己写的一些delegate,我们会用weak,而不是assign,weak的好处是当对应的对象被回收时,指针也会自动被设置为nil。

  • 循环未结束

    如果某个ViewController中有无限循环,也会导致即使ViewController对应的view关掉了,ViewController也不能被释放。
    这种问题常发生于animation处理。

    CATransition *transition = [CATransition animation];
    transition.duration = 0.5;
    tansition.repeatCount = HUGE_VALL;
    [self.view.layer addAnimation:transition forKey:"myAnimation"];
    

上例中,animation重复次数设成HUGE_VALL,一个很大的数值,基本上等于无限循环了。
解决办法是,在ViewController关掉的时候,停止这个animation。
-(void)viewWillDisappear:(BOOL)animated {
[self.view.layer removeAllAnimations];
}

  • ** try...catch 的使用**
    但如果 doSomethingMayThrowException 方法抛出了异常,那么 object 对象就无法释放。如果 object 对象持有了重要且稀缺的资源,就可能会造成严重后果。

PS其他需要注意的问题

大次数循环内存暴涨问题

记得有道比较经典的面试题,查看如下代码有何问题:

    for (int i = 0; i < 100000; i++) {
        NSString *string = @"Abc";
        string = [string lowercaseString];
        string = [string stringByAppendingString:@"xyz"];
        NSLog(@"%@", string);
}

该循环内产生大量的临时对象,直至循环结束才释放,可能导致内存泄漏,解决方法为在循环中创建自己的autoReleasePool,及时释放占用内存大的临时变量,减少内存占用峰值。

for (int i = 0; i < 100000; i++) {
        @autoreleasepool {
            NSString *string = @"Abc";
            string = [string lowercaseString];
            string = [string stringByAppendingString:@"xyz"];
            NSLog(@"%@", string);
      }
  }

附、如何检测App的内存泄漏问题

  • 借助Xcode自带的Instruments工具(选取真机测试

    Instruments
  • 简单暴力的重写dealloc方法,加入断点或打印判断某类是否正常释放。

    dealloc
  • 使用Xcode8中自带的有内存检测警告。


  • 通过Facebook出品的FBMemoryProfiler工具类进行检测.

集成后的显示

</br>

这篇ARC下的内存泄漏,洋洋洒洒说了这么多,算是总结的比较详细和全面的。希望对大家有价值。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容