iOS自问自答:总结内存管理与优化

目录

  1. ARC下如何避免内存泄露?如何检测?
  2. 你是如何做内存优化的?
  3. __block你知道多少?在什么时候使用?
  4. 关于防止APP崩溃你做了哪些努力?
  5. 你是如何做线上Bug定位的?
  6. 关于经验和技巧还有什么想说的?

1. ARC下如何避免内存泄露?如何检测?

  • 避免:

    • 注意使用block时是否造成循环引用,使用__weak 配合 __strong关键字打破闭环

      不是所有的block都要避免循环引用。所谓“循环引用”,指的是双向的强引用(即self强引用block,block也强引用self),单向的强引用则不用担心,系统的某些block api(如UIView的block版本写动画)或者第三方库如MJRefreshSDWebImage等未形成闭环均不用考虑循环引用问题。

    • delegate使用weak声明比assign好,因为使用weakdelegate成员变量会在持有者销毁时自动被赋为nil对象回收时Weak指针自动被置为nil的实现原理,典型应用:一句话移除所有通知 [[NSNotificationCenter defaultCenter] removeObserver:self];),此时向空对象发消息objc_msgSend(obj, @selector(methodName:)判断obj为 nil 则selector也为 nil 从而直接返回 0(nil) 而不会引起crash
    • 注意CoreFoundation对象的使用,使用完成后记得主动调用相应的CFRelease()方法
    • 例如NSTimer加入到Runloop中,界面消失时记得将定时器销毁,建议使用NSTimer的分类(如YYKit的NSTimer+YYAdd),并在dealloc中调用[timer invalidate]停止定时器
  • 检测:

检测代码中是否存在循环引用问题,可使用 Facebook 开源的一个检测工具FBRetainCycleDetector,这里有两篇很棒的文章翻译并介绍了它的相关用法:
[译文]在iOS上自动检测内存泄露
FBMemoryProfiler 基础教程

  • 使用Xcode -> Product -> Analyze 分析memory警告,可以发现局部变量忘记release的情况(或者申请了内存却未使用)
  • 在Xcode -> Debug area中点击Debug Memory GraphDebug View Hierarchy按钮,在Debug navigator区查看紫色感叹号情况,缺点是每个屏幕都要点击一下
    Debug area
  • 使用Instruments工具之Leaks检测,申请了内存然而没有指向这块内存的指针存在则可以认为是leak了
  • 使用Instruments工具之Allocations的mark heap,多次重复操作标记区内存增加应该为0
  • 部分循环引用的地方Leaks检测不出来,提供一个思路,结合runtime特性AOP编程,在所有的dealloc方法中打印描述信息,若发现退出时不打印,显然是被哪个对象持有了,再仔细排查

2. 你是如何做内存优化的?

先弄清楚这里的学问,再来谈 iOS 内存管理与优化(二)

  • Weak-Strong-Dance防止block和对象间的循环引用(思维延展:多线程下并不一定安全,应对strongSelf进行nil检测 ,Weak-Strong-Dance真的安全吗?

    贴一下@weakify,@strongify的宏定义写法

  • 降低内存峰值(reuse,cache,lazy-loading,async就不用多说了)(注:关于属性strong定义的UIView及其子类,在调用removeFromSuperView后并不会立即释放,确定不再使用后可手动置nil)

  • 图片加载方式:带缓存imageNamed,不带缓存contentsOfFile(适用于大图片)
    使用SDWebImage和YYImage下载高分辨率图,导致内存暴增的解决办法
    另外加载GIF图,尤其容易导致达到内存峰值从而引起闪退,可采用FLAnimatedImage解决方案

  • 采用图片拼合(多张图片打包整合到一张大图片上显示,cocos2d采用此技术使用OpenGL显示图片,优势:内存使用、载入时间、渲染性能等等)方式,使用时利用CALayercontentsRect属性裁剪指定位置的图片,来源iOS核心动画:寄宿图

  • 在for循环中主动添加@autoreleasepool{},保证在每次迭代完成后释放临时变量所占内存(注:当retainCount=1且将要release时,打印结果其值不会变为0,原因是为了节省对象引用计数-1的开销,而会在稍后某个时间点或内存不足时释放)

  • 避免滥用单例对象。单例会一直持有资源,合理使用其他替换方案

  • 减少NSDateFormatterNSCalendar初始化次数(多次使用模仿单例写法static+dispatch_once,但更好的方案是使用时间戳如- (NSDate*)dateFromUnixTimestamp:(NSTimeInterval)timestamp { return [NSDate dateWithTimeIntervalSince1970:timestamp]; }

  • 处理内存警告,释放不再使用的资源

    - (void)didReceiveMemoryWarning {
      [super didReceiveMemoryWarning];//即使没有显示在window上,也不会自动的将self.view释放。注意跟ios6.0之前的区分
      // Add code to clean up any of your own resources that are no longer necessary.
      // 此处做兼容处理需要加上ios6.0的宏开关,保证是在6.0下使用的,6.0以前屏蔽以下代码,否则会在下面使用self.view时自动加载viewDidUnLoad
      if ([[UIDevice currentDevice].systemVersion floatValue] >= 6.0) {
        //需要注意的是self.isViewLoaded是必不可少的,其他方式访问视图会导致它加载,在WWDC视频也忽视这一点。
        if (self.isViewLoaded && !self.view.window)// 是否是正在使用的视图
        {
            // Add code to preserve data stored in the views that might be needed later.
            // Add code to clean up other strong references to the view in the view hierarchy.
            self.view = nil;// 目的是再次进入时能够重新加载调用viewDidLoad函数。
        }
      }
    }
    

3. __block你知道多少?在什么时候使用?

  • 本身不能避免block内部循环引用,可在block内部将 blockObj 置为 nil 的方式避免循环引用
  • 提升了变量的作用域,可以在block内部修改局部变量(通过block进行闭包的变量是const的)

block不允许修改外部变量的值,这里所说的外部变量的值,指的是栈中指针的内存地址。__block所起到的作用就是只要观察到该变量被 block 所持有,就将“外部变量”在栈中的内存地址放到了堆中。进而在block内部也可以修改外部变量的值。

  • 对于变量是指针及数组,只复制了指针,两个指针指向同一个(堆)地址
  • __block__weak的区别
关键字 适用模式 修饰类型 特性
__block ARC、MRC 对象、基本数据类型(int) 可在block块中被重新赋值
__weak ARC 对象(NSString) 对象回收时自动被置为nil

for循环+block块嵌套(常用于多张图片上传及下载)时,可保证block块内部执行完毕后才进入下一次循环,因为实际修改的是同一个地址的内容

一个同步访问数据的栗子
__block NSDictionary *dict = nil; 
do { 
    @autoreleasepool {
        NSMutableURLRequest *req = [[NSMutableURLRequest alloc]initWithURL:[NSURL URLWithString:@""]];
        req.timeoutInterval = 10;   // timeoutInterval has no effect

        dispatch_semaphore_t semaphore = dispatch_semaphore_create(0);
    
        __block NSURLSessionTask *dataTask = [[NSURLSession sharedSession] dataTaskWithRequest:req completionHandler:^(NSData * _Nullable data, NSURLResponse * _Nullable response, NSError * _Nullable error) {
            if (!error && data) {
               dict = [NSJSONSerialization JSONObjectWithData:data options:NSJSONReadingMutableContainers|NSJSONReadingMutableLeaves error:nil];
            } else {
               NSLog(@"error: %@",[error description]);
            }
            dispatch_semaphore_signal(semaphore);
        }];
        [dataTask resume];
    
        dispatch_semaphore_wait(semaphore, DISPATCH_TIME_FOREVER);
    }
} while (nil == dict);

4. 关于防止APP崩溃你做了哪些努力?

APP发生崩溃常出于以下这些情况:

  1. 找不到对应的方法unrecognized selector sent to instance
  2. 容器越界:数组越界/字典设置空对象等
  3. 访问了僵尸对象EXC_BAD_ACCESS

解决方案如下:

  1. 利用method-swizzling覆盖消息转发链关键方法,具体做法可以参考这篇文章(iOS 防止应用崩溃解决方案,这个项目也不错AvoidCrash),不再赘述

  2. 同样是用runtime黑魔法替换方法实现,替换潜在崩溃的方法实现,如增加NSArray+Safe分类

     + (void)load {
       static dispatch_once_t onceToken;
       dispatch_once(&onceToken, ^{
           [objc_getClass("__NSArrayI") swizzleSelector:@selector(objectAtIndex:) withSwizzledSelector:@selector(safeObjectAtIndex:)];
       });
     }
    
     - (id)safeObjectAtIndex:(NSUInteger)index {
       // 数组越界也不会崩,但是开发的时候并不知道数组越界
       if (index > (self.count - 1)) { // 数组越界
           NSLog(@"数组越界了: index = %ld, array = %@", index, self);
           NSAssert(1, @"数组越界了"); // 只有开发的时候才会造成程序崩了
           return nil;
       } else { // 没有越界
           return [self safeObjectAtIndex:index];
       }
     }
    
  3. 在Xcode中启用僵尸调试模式Zombie Objects

僵尸对象的工作原理是系统即将回收的对象转化为僵尸对象而不彻底回收,在运行时创建一个_NSZombie_+原类名的新类,对象的isa指针会被修改指向这个僵尸类,在消息转发机制中___forwarding___总是会先检查接收消息的对象所属的类名,一旦发现前缀为_NSZombie_,则会特殊处理。

通常发生EXC_BAD_ACCESS,需要检查是否有在dealloc中移除通知及KVO(iOS 9以下系统需要手动移除),检查属性关键字是否设置正确,此外还有delegate方法是否判断了事件的响应者等。

虽然有这些解决办法,但是问题依然存在,必然会导致其他问题,crash本来就是帮助开发者找到问题并及时修复,过分的追求减少崩溃率除了保持KPI并不能带来体验上的提升,开发更多的还是要完善容错处理,写出更健壮的代码。

5. 你是如何做线上Bug定位的?

iOS 崩溃日志分析
iOS调试之 crash log分析
iOS 应用Crash日志分析整理

  • 通过第三方Fabric、Bugly、友盟等SDK上传

    缺点:因为内存占用过大被看门狗杀掉的无法定位出错点,诸如一些野指针问题也没有发现

  • 通过Apple后台搜集,可在Xcode -> Window -> Organizer ->Crashes中直接查看定位

    缺点:用户设置不上传诊断信息就看不到日志了

  • 分析iOS Crash文件:符号化iOS Crash文件的3种方法
Require Location
分析工具symbolicatecrash 打开终端find /Applications/Xcode.app -name symbolicatecrash -type f找到工具所在位置 /Applications/Xcode.app/Contents/SharedFrameworks/DVTFoundation.framework/Versions/A/Resources/symbolicatecrash
.dSYM符号文件 Xcode -> Window -> Organizer -> Archives -> 选中项目 + Download dSYMs… /右键Show in Finder -> .xcarchive+右键Show in Finder
Xcode -> Products -> .app + 右键Show in Finder
crash报告 Xcode -> Window -> Device -> 选中测试手机 -> View Device Logs
Finder前往文件夹 -> ~/Library/Logs/CrashReporter/MobileDevice/<DEVICE_NAME>
Optional
.app文件 release生成的.ipa文件后缀改为.zip -> 解压 -> Payload目录下的appName.app文件
终端解析命令: 要求三者在同一文件夹下
  • 先执行export DEVELOPER_DIR=/Applications/Xcode.app/Contents/Developer,防止出现Error:"DEVELOPER_DIR" is not defined at ./symbolicatecrash line 60.
  • cd到同一文件夹下,执行./symbolicatecrash ./*.crash ./*.app.dSYM > symbol.crash

6. 关于经验和技巧还有什么想说的?

都放在github上了 MyLearningCorner,持续更新ing~

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

推荐阅读更多精彩内容

  • *面试心声:其实这些题本人都没怎么背,但是在上海 两周半 面了大约10家 收到差不多3个offer,总结起来就是把...
    Dove_iOS阅读 27,131评论 30 470
  • iOS开发中, 之前一直使用swift, 因此对于Objective-C的内存管理机制长期处于混乱的一知半解状态....
    icetime17阅读 839评论 1 8
  • 禅与 Objective-C 编程艺术 (Zen and the Art of the Objective-C C...
    GrayLand阅读 1,607评论 1 10
  • 北书房阅读 539评论 6 13
  • 到车站时下了出租车双脚落地,感觉走起路来十分异样时,她才发现自己居然穿了两只不一样的鞋。鞋的材质是一样的,都是黑色...
    莫莫queen阅读 374评论 0 0