美文网首页
编码篇-ARC下的内存泄漏

编码篇-ARC下的内存泄漏

作者: 進无尽 | 来源:发表于2017-07-18 16:10 被阅读0次

</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下的内存泄漏,洋洋洒洒说了这么多,算是总结的比较详细和全面的。希望对大家有价值。

相关文章

  • 编码篇-ARC下的内存泄漏

    前言 内存泄露是一个相对挺严重的问题,可是它的存在未引起足够的重视,如果程序运行时一直分配内存而不及时释放无用的内...

  • ARC下的内存泄漏

    ARC下的内存泄漏 ARC全称叫 ARC(Automatic Reference Counting)。在编译期间,...

  • 轻松搞定内存泄露的产生,检测和解决

    以前MRC的下由于缺少release等会经常造成内存泄漏问题,如今在ARC下内存泄漏的问题已经减少了很多,但是还有...

  • 2018-04-24

    第五章:内存管理 29. 什么是ARC:对象被引用的次数。 30. ARC 下可以用“静态分析器”查找内存泄漏问题...

  • iOS常见面试题

    一、ARC下怎么检查内存泄漏,你遇到最多的内存泄漏原因是什么? 无固定答案,主要是看她是否遇到过这个问题。 二、如...

  • 关于app的性能优化

    1.使用arc管理内存,避免内存泄漏。 2.给UITableViewCells,UICollectionViewC...

  • AFN 引起的内存泄漏

    今天在做内存泄漏检测时,发现很多地方都有内存泄漏,而引起泄漏的原因居然AFN, 网上查了资料了解到由于ARC机制导...

  • 性能优化——内存泄漏(3)代码分析篇

    内存泄漏系列文章:性能优化——内存泄漏(1)入门篇性能优化——内存泄漏(2)工具分析篇性能优化——内存泄漏(3)代...

  • 性能优化——内存泄漏(2)工具分析篇

    内存泄漏系列文章:性能优化——内存泄漏(1)入门篇性能优化——内存泄漏(2)工具分析篇性能优化——内存泄漏(3)代...

  • 性能优化——内存泄漏(1)入门篇

    内存泄漏系列文章:性能优化——内存泄漏(1)入门篇性能优化——内存泄漏(2)工具分析篇性能优化——内存泄漏(3)代...

网友评论

      本文标题:编码篇-ARC下的内存泄漏

      本文链接:https://www.haomeiwen.com/subject/intqkxtx.html