iOS 内存警告的处理

更多内容请挪步我的博客

[LEAF Photo] 中要处理图片,如果操作时间长的话会收到 Memory Warning,内存处理的不好的话就会崩溃,下面记录下处理内存警告的一些优化。

处理内存警告

处理所有 ViewController 中的 didReceiveMemoryWarning 方法
didReceiveMemoryWarning 方法中应当把缓存的变量都清空,这些变量最好都是通过懒加载的方式创建,这样收到内存警告后也会自动加载。

@interface TestViewController () 
@property (strong, nonatomic) NSMutableArray *dataArray;
- (void)didReceiveMemoryWarning {
    [super didReceiveMemoryWarning];
    // Dispose of any resources that can be recreated.
    self.dataArray = nil;
}
- (NSMutableArray *)dataArray {
    if (!_dataArray) {
        _dataArray = [[NSMutableArray alloc] init];
    }
    // 赋值处理 ... 
    return _dataArray;
}

在处理图片的 Controller 中当用户点击某个按钮时会加载一些 View,例如处理文字、背景、绘图笔等是弹出的设置页面,这些 View 都封装成了自定义的类从 Controller 中分离出去,这些类的内存可以被处理,如果在 didReceiveMemoryWarning 中把这些视图清理时,需要判断当前视图是否是否正在使用,否则可能出现正在操作这些视图时,由于收到内存警告而被删除导致错误,在 iOS 6 以上的版本中可以用如下方式判断视图是否在使用

- (void)didReceiveMemoryWarning {
    [super didReceiveMemoryWarning];
    if ([self.view window] == nil)   { // 视图是否正在使用
       self.settingView = nil;
    }
}

另外,NSCache 缓存类会在收到 Memory Warning 时自动删除缓存内容,不需要手动做清理,有个 Demo 点击查看

还可以在 AppDelegate 中实现 applicationDidReceiveMemoryWarning 方法做一些全局数据清理

- (void)applicationDidReceiveMemoryWarning:(UIApplication *)application {
    
}

消耗内存的对象创建放到 AutoReleasePool 中

函数中出现很多中间变量占据大量内存,或者不多的中间变量但是也是占用较大内存的,需要放到自动释放池中

   @autoreleasepool {
        ALAsset *asset = self.assetArray[i];
        if (asset && ![asset isEqual:[NSNull null]]) {
            UIImage *img = [UIImage imageWithCGImage:[[asset defaultRepresentation] fullScreenImage]];
            // ...
        }
    }

使用 Allocations 工具查看是否有应该释放但是没有被释放的内存

如果进入一个页面后退出该页面,内存没有降回或者接近降回到进入该页面前的内存数,说明该页面有内存没有被释放。

例如点击 [PhotoBook] 中点击添加照片的按钮,添加照片后退出这个页面后发现内存中驻留了该页面的内存,而该段内存是存在 LTPhotoPickerViewController 的某个变量中的,查看页面代码发现添加图片按钮中有如下代码,每次点击都创建了相同的控制器,但是没有对其释放的地方,所以创建了多个重复的对象,而这个对象中保存了很多内存。

- (void)stickerPhotoAction:(id)sender {
    LTPhotoPickerViewController *pickerController = [[LTPhotoPickerViewController alloc] init];
    [self.navigationController pushViewController:pickerController animated:YES];
}

这种简单的错误很容易修改,只是早期的时候脑袋坏掉不小心写错,要解决问题最困难的问题不是如何修改,而是如何找到问题在哪。

修改方法:将 pickerController 改为属性,每次点击按钮的时候使用同样的属性对象即可。

循环引用

除了 Block 内部要注意不要引用自己,还要注意是否有属性应当是 weak 的,但是设置成 strong,导致内部引用计数错误导致的无法释放,这种现象也要靠 Allocations 工具检查,页面退出后还有对象被 Persistent,可以看看是否有该 weak 的被 strong 了。

如果使用照片尽量对其进行压缩节省内存

// ALAsset *asset = ...;
UIImage *img = [UIImage imageWithCGImage:[[asset defaultRepresentation] fullScreenImage]]; 
NSData *imgData = UIImageJPEGRepresentation(fullScreenImage, 0.7);
UIImage *img = [UIImage imageWithData:imgData];
// scaled
UIImage *img = [UIImage imageWithCGImage:[[asset defaultRepresentation] fullScreenImage] scale:scale orientation:UIImageOrientationUp];

同样是获取相册中的照片,两种方式进行压缩节省内存

图片保存时注意保存图片的大小

如果需要自己创建图片上下文并绘制,给用户选择的图片大小最大不要超过 4096 ** 4096 像素,可以给用户大、中、小几种选择。

创建图片上下文使用 UIGraphicsBeginImageContextWithOptions(size, YES, [UIScreen mainScreen].scale); 的话,在主流机上 scale 都是 2,那么创建出的图片像素都是 size 大小的 2倍,如果让用户选择 3200 ** 3200 大小的照片,保存后就是 6400 ** 6400,图片这么大很容易造成内存吃紧,收到警告,所以不要使用 UIGraphicsBeginImageContextWithOptions 方式,保存图片后看下图片是否是想要的大小。

GCD Timer 的使用注意事项

NSTimer 一定要在恰当的地方执行 invalidate,否则会造成内存泄漏,但是用 GCD Timer 就可以绕过这个问题。但是在 [PhotoBook] 项目中添加 GCD Timer 后想要在某种情况下,把 Timer 暂停,于是设置了某个变量,当该变量在某种情况下将 timer 暂停,于是有了下面的代码

dispatch_source_set_event_handler(timer, ^() {
    // if (condition)  {      
        // dispatch_suspend(timer);
   // }
}
dispatch_resume(timer);

发现在 dispatch_source_set_event_handler 中加上 dispatch_suspend,该页面无法被释放。具体分析请挪步这里

参考内容

25 iOS App Performance Tips Tricks

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

推荐阅读更多精彩内容

  • 内存管理 简述OC中内存管理机制。与retain配对使用的方法是dealloc还是release,为什么?需要与a...
    丶逐渐阅读 1,946评论 1 16
  • IPhone下每个app可用的内存是被限制的,如果一个app使用的内存超过20M,则系统会向该app发送Memor...
    Yumazhiyao阅读 3,613评论 1 11
  • 37.cocoa内存管理规则 1)当你使用new,alloc或copy方法创建一个对象时,该对象的保留计数器值为1...
    如风家的秘密阅读 827评论 0 4
  • *面试心声:其实这些题本人都没怎么背,但是在上海 两周半 面了大约10家 收到差不多3个offer,总结起来就是把...
    Dove_iOS阅读 27,121评论 29 470
  • 秋意斑驳,落叶双双, 醉了流年 夕阳残影,凄风瑟瑟, 乱了青丝 脚步渐缓,眸定云远去 南归的鸿雁,敲开尘封的记忆 ...
    亮子_4ae7阅读 86评论 0 0