APP性能监控之FOOM检测实践

原文:橘子不酸丶
转载:https://juejin.im/post/5e154ad35188253aad5c7c37

前言

iOS开发中 随着应用在版本迭代过程中,APP大小逐渐增大,APP占用内存也随着功能的增加而越来越多,因此APP在线上发生FOOM(Foreground Out Of Memory)的频率也会越来越高。

一般对于用户而言,发生FOOM时和crash表现一样,然而在通常在线上crash收集过程中FOOM却并不能被收集到,所以如何收集应用在线上FOOM发生的频率,以及FOOM发生时内存占用情况也是我们的一个重要的监控指标。

原理

Facebook早在2015年8月提出FOOM检测办法,大致原理是排除各种情况后,剩余的情况是FOOM,Reducing FOOMs in the Facebook iOS app
本文则基于这种检测办法来具体分析和实现一次OOM发生后的区分以及上报。

image

怎么判断一次OOM

  1. APP版本升级
  2. 调用Exit()或者Abort()退出应用
  3. APP发生Crash
  4. 用户通过Home键强制退出APP
  5. 手机系统升级引起的重启
  6. APP在后台BOOM!
  7. APP在前台FOOM!

实现

1.启动检测

当APP启动时,我们需要尽可能早的在application:didFinishLaunchingWithOptions:之后开启我们的OOM检测。

typedef void (^MJYPOutOfMemoryEventHandler)(BOOL wasInForeground);
- (void)beginMonitoringMemoryEventsWithHandler:(nonnull MJYPOutOfMemoryEventHandler)handler;

2.APPVersion以及OSVersion记录

获取当前APP版本号并取出上次存储的APP版本号进行对比

  NSDictionary *infoDictionary = [[NSBundle mainBundle] infoDictionary];
  NSString *majorVersion = infoDictionary[@"CFBundleShortVersionString"];
  NSString *minorVersion = infoDictionary[@"CFBundleVersion"];
  return [NSString stringWithFormat:@"%@.%@", majorVersion, minorVersion];

获取当前OSVersion并取出上次存储的OSVersion进行对比

  [NSString stringWithFormat:@"%@.%@.%@", @(version.majorVersion), @(version.minorVersion), @(version.patchVersion)]

3.Terminate以及前后台切换监听

监听terminate和前后台切换之后,我们需要处理用户强制退出程序也就是terminate的情况,以及还需要来记录和区分前后台的状态,当发生OOM时我们则可以通过此状态来区分FOOM。

[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(applicationWillTerminate:) name:UIApplicationWillTerminateNotification object:nil];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(applicationDidEnterBackground:) name:UIApplicationDidEnterBackgroundNotification object:nil];
[[NSNotificationCenter defaultCenter] addObserver:self selector:@selector(applicationWillEnterForeground:) name:UIApplicationWillEnterForegroundNotification object:nil];

4.Hook exit()以及abort()方法监听程序退出

abort() : 立即结束,不做任何操作。

exit() : 释放所有静态全局的对象、缓存,关掉所有的I/O通道,然后终止程序。如果有函数通过atexit来注册,还会调用注册的函数,如果atexit函数抛出异常就会直接调用terminate。

atexit() : 注册程序正常终止时要被调用的函数,一个进程可以登记多达32个函数,这些函数将由exit自动调用,通常这32个函数被称为终止处理程序,并调用atexit函数来登记这些函数。

我们可以通过fishhook来hook C中的这两个退出程序的函数。

fishhook是Facebook提供的一个动态修改链接mach-O文件的工具。利用MachO文件加载原理,通过修改懒加载表(Lazy Symbol Pointers)和非懒加载表(Non-Lazy Symbol Pointers)这两个表的指针达到C函数HOOK的目的。fishhook的具体原理可以参考Github源码以及简介。

具体做法如下:

static void (*orig_exit)(int);
static void (*_orig_exit)(int);
static void (*orig_abort)(void);
void my_exit(int value) {
    [[MJYPOutOfMemoryMonitor sharedInstance] appDidExit];
    orig_exit(value);
}
void _my_exit(int value) {
    [[MJYPOutOfMemoryMonitor sharedInstance] appDidExit];
    _orig_exit(value);
}
void my_abort(void) {
    [[MJYPOutOfMemoryMonitor sharedInstance] appDidExit];
    orig_abort();
}
+ (void)load {
  static dispatch_once_t onceToken;
  dispatch_once(&onceToken, ^{
      rebind_symbols((struct rebinding[1]){
          {"_exit", (void *)_my_exit, (void *)&_orig_exit}
      }, 1);
      rebind_symbols((struct rebinding[1]){
          {"exit", (void *)my_exit, (void **)&orig_exit}
      }, 1);
      rebind_symbols((struct rebinding[1]){
          {"abort", (void *)my_abort, (void **)&orig_abort}
      }, 1);
  });
}

5.用户Crash过滤

一般情况下,我们可以在收集到用户Crash发生时,来处理过滤Crash的情景。
比如说我们使用Tencent的Bugly库时,我们可以在Bugly回调attachmentForException:时进行调用过滤。
[[MJYPOutOfMemoryMonitor sharedInstance] appDidCrash];//OOM监控调用

6.记录OOM上报

当APP启动时,对上述情况进行判断,排除以上情况之后,我们则认为APP在上一次运行过程中发生了FOOM。此时则上报上一次发生FOOM时的页面信息场景以及内存对象分配。

其他误判情况

前台卡死引起系统watchdog强杀,通常原因是前台线程过多或者发生死锁,或CPU使用率持续过高等,这类强杀无法被App捕获,需要结合卡顿监控来加以区分。

后续

单独的知道我们上一次APP运行过程中发生FOOM后,我们可以知道我们应用的线上运行内存稳定性,后续则需要完善在FOOM场景发生时的内存堆栈数据和内存对象分配,以及页面信息记录。附FOOM Report Demo

参考资料

iOS微信内存监控 - 杨津

OOMDetector组件 - 手Q团队

FBRetainCycleDetector
FBMemoryProfiler
FBAllocationTracker

Reducing FOOMs in the Facebook iOS app

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

推荐阅读更多精彩内容

  • 欢迎大家前往云+社区,获取更多腾讯海量技术实践干货哦~ 作者:杨津,腾讯移动客户端开发 高级工程师 由WeTest...
    yuguang1阅读 486评论 0 0
  • iOS微信内存监控 云加社区 2018-03-02 10:45:31 更多腾讯海量技术文章,请关注云+社区:htt...
    树懒啊树懒阅读 756评论 1 5
  • 以下为文章正文,如果觉得有用,欢迎给她打赏。 为了能够第一时间发现程序问题,应用程序需要实现自己的崩溃日志收集服务...
    赤色追风阅读 2,523评论 1 11
  • 相信最近因为疫情,很多人足不出户,特别像我这种身处湖北境内的,人人自危。 今天一觉醒来,又有了新的坏消息。科比坠机...
    珩玥阅读 1,282评论 11 26
  • 01 近段时间的早上,在我家,都会上演一段可歌可泣的“小别离”。 娃扯着嗓子,抱着她爸大腿,可怜兮兮地哀求,“爸爸...
    毅行生长阅读 564评论 0 5