iOS中的性能监控方法

APP开发中性能问题无疑是很重要的一点,有几项指标可以看出APP的性能是否存在问题,内存使用量,FPS,以及CPU使用率。在开发阶段这些数据的测试很容易,有一些在Xcode编译项目时就会有显示,而剩下的则可以使用Instruments来进行监控。可以说在线下的监控Instruments为我们考虑到了方方面面,但是光是线下监控是完全不够的,因为APP上线后运行的环境是十分复杂的,我们在测试时不可能模拟的面面俱到,因此就需要针对线上的性能问题进行监控。

内存使用量

内存使用是很重要的一点,如果我们的内存使用量过大,APP就会被系统杀掉,给用户的表现就是闪退,这是很严重的一个问题,而我们的APP如果被系统强杀会产生一个叫jetsam的日志,这个日志可以通过手机中设置 -> 隐私 -> 分析中看到相关日志。

现代的进程在虚拟内存中的运行是以分页形式存在的,这样做可以节省内存空间,因为APP在运行的时候只有一部分会映射到虚拟内存中,而不是整个APP都会被加载到虚拟内存上,只有使用到的部分才会被映射,而jetsam日志就是以页数为单位来衡量一个APP使用的内存是否超过限制。

"rpages" : 89600,
"reason" : "per-process-limit",

像这样,表明我们使用了89600个内存页,超出了单进程的内存限制,如果可以知道一页的大小就可以知道系统对单个进程的内存限制是多少。注意这个限制不是固定的,而是系统根据当前内存情况来决定的

jetsam日志

可以看到一页的大小是16384,这样就可以计算出当前 App 的内存限制值:pageSize * rpages / 1024 /1024 =16384 * 89600 / 1024 / 1024 得到的值是 1400 MB,即 1.4G。

iOS系统会使用一个优先级最高的线程vm_pressure_monitor来监控系统内存压力的情况,并通过一个堆栈来维护所有的APP进程,如果发现某个进程的内存快要超出限制了就会发出通知,内存有压力的APP就会执行代理,也就是熟悉的didReceiveMemoryWarning,在这里面可以写一个释放内存的方法,这是最后的机会去避免APP被强杀。

不过很遗憾APP在上限后我们是无法去获取jetsam日志的,这属于每个用户的隐私,我们的权限是无法获得的,但是iOS还为我们提供了其他的方法去获取内存的使用情况,以供我们在APP内存收到警告时查看当前内存的使用情况。

我们可以写一个文件导入如下头文件

#import <mach/mach.h>

从头文件的名称就可以看出这个是系统级的方法,iOS系统提供了一个函数task_info可以获得当前进程的使用情况

struct mach_task_basic_info info;
mach_msg_type_number_t size = sizeof(info);
kern_return_t kl = task_info(mach_task_self(), MACH_TASK_BASIC_INFO, (task_info_t)&info, &size);
float used_mem = info.resident_size;
NSLog(@"使用了 %f MB 内存", used_mem / 1024.0f / 1024.0f)

MACH_TASK_BASIC_INFO这个结构体中定义了一系列和内存有关的变量

struct mach_task_basic_info {
    mach_vm_size_t  virtual_size;       /* virtual memory size (bytes) */
    mach_vm_size_t  resident_size;      /* resident memory size (bytes) */
    mach_vm_size_t  resident_size_max;  /* maximum resident memory size (bytes) */
    time_value_t    user_time;          /* total user run time for
                                         *  terminated threads */
    time_value_t    system_time;        /* total system run time for
                                         *  terminated threads */
    policy_t        policy;             /* default policy for new threads */
    integer_t       suspend_count;      /* suspend count for task */
};

但是这样测出来和Xcode中实际显示的出入较大,后来苹果的开发者大会说task_vm_info结构体中的phys_footprint才是真正的物理内存使用量


struct task_vm_info {
  mach_vm_size_t  virtual_size;       // 虚拟内存大小
  integer_t region_count;             // 内存区域的数量
  integer_t page_size;
  mach_vm_size_t  resident_size;      // 驻留内存大小
  mach_vm_size_t  resident_size_peak; // 驻留内存峰值

  ...

  /* added for rev1 */
  mach_vm_size_t  phys_footprint;     // 物理内存

  ...

由此就可以写一个简单的内存使用量的检测方法

- (float)getMemoryUse{
    //TASK_VM_INFO中存储物理内存使用信息
    int64_t memoryUsageInByte = 0;
    task_vm_info_data_t vmInfo;
    mach_msg_type_number_t count = TASK_VM_INFO_COUNT;
    kern_return_t kernReturn = task_info(mach_task_self(), MACH_TASK_BASIC_INFO, (task_info_t)&info, &size);
    if (kernReturn != KERN_SUCCESS) { return NSNotFound; }
    memoryUsageInByte = (int64_t) vmInfo.phys_footprint;
    return memoryUsageInByte/1024.0/1024.0;
}

为了测试我写了一个方法,每隔一秒生成1000个对象装入类中的可变数组,保证其不会被释放,测试内存使用量。

- (void)startMonitor{
    dispatch_queue_t queue = dispatch_get_global_queue(DISPATCH_QUEUE_PRIORITY_DEFAULT, 0);
    //创建一个定时器(dispatch_source_t本质上还是一个OC对象)
    self.timer = dispatch_source_create(DISPATCH_SOURCE_TYPE_TIMER, 0, 0, queue);
    
    //设置定时器的各种属性
    dispatch_time_t start = dispatch_time(DISPATCH_TIME_NOW, (int64_t)(0*NSEC_PER_SEC));
    uint64_t interval = (uint64_t)(1.0*NSEC_PER_SEC);
    dispatch_source_set_timer(self.timer, start, interval, 0);
    
    
    //设置回调
    __weak typeof(self) weakSelf = self;
    dispatch_source_set_event_handler(self.timer, ^{
        //定时器需要执行的操作
        self->usedMemory = [[BZMemoryMonitor shareInstance] getMemoryUse];
        [weakSelf increaseMemory];
        dispatch_async(dispatch_get_main_queue(), ^(void){
            //Run UI Updates
            weakSelf.useLabel.text = [NSString stringWithFormat:@"使用内存:%f",self->usedMemory];
        });
       
    });
    //启动定时器(默认是暂停)
    dispatch_resume(self.timer);
}


- (void)increaseMemory{
    
    for (int i = 0; i < 1000; i++) {
        NSObject *obj = [[NSObject alloc] init];
        [self.array addObject:obj];
    }
}

工具使用效果如下:


内存使用获取

和Xcode中显示的内存使用基本一致。

FPS监控

提到FPS监控很多人可能都会知道使用CADisplayLink,什么是CADisplayLink呢?

CADisplayLink是CoreAnimation提供的另一个类似于NSTimer的类,它总是在屏幕完成一次更新之前启动,它的接口设计的和NSTimer很类似,所以它实际上就是一个内置实现的替代,但是和timeInterval以秒为单位不同,CADisplayLink有一个整型的frameInterval属性,指定了间隔多少帧之后才执行。默认值是1,意味着每次屏幕更新之前都会执行一次。但是如果动画的代码执行起来超过了六十分之一秒,你可以指定frameInterval为2,就是说动画每隔一帧执行一次(一秒钟30帧)或者3,也就是一秒钟20次,等等。

CADisplayLink可以以屏幕刷新的频率调用指定selector,而且iOS系统中正常的屏幕刷新率为60Hz(60次每秒),那只要在这个方法里面统计每秒这个方法执行的次数,通过次数/时间就可以得出当前屏幕的刷新率了。

由此可以写出一个FPS监控的工具:

- (void)setupDisplayLink{
    // 初始化CADisplayLink
    _displayLink = [CADisplayLink displayLinkWithTarget:self selector:@selector(fpsCount:)];
    // 把CADisplayLink对象加入runloop
    [_displayLink addToRunLoop:[NSRunLoop mainRunLoop] forMode:NSRunLoopCommonModes];
}


// 方法执行帧率和屏幕刷新率保持一致
- (void)fpsCount:(CADisplayLink *)displayLink {
    if (_lastTimestamp == 0) {
        _lastTimestamp = self.displayLink.timestamp;
    } else {
        _performTimes++;
        // 开始渲染时间与上次渲染时间差值
        NSTimeInterval useTime = self.displayLink.timestamp - _lastTimestamp;
        if (useTime < 1) return;
        _lastTimestamp = self.displayLink.timestamp;
        // fps 计算
        float fps = _performTimes / useTime;
        NSLog(@"%f",fps);
        _performTimes = 0;
    }
}

但是我发现这种方式与Instrument检测出来的差距有些大,我的demo中滑动已经十分卡顿,但是这个工具依旧显示fps在57左右,后来参考这篇文章才发现其原因。

引用文章内容:
iOS中每一帧画面的生成是一个复杂的过程,但简单来说需要经过以下步骤:

1、系统根据你的代码,设置布局各个元素的位置(frame、AutoLayout)、属性(颜色、透明度、阴影等)。
2、CPU对需要提前绘制的元素、图形使用Core Graphics进行绘制。
3、CPU将一切需要绘制到屏幕上的内容(包括解压后的图片)打包发送到GPU
4、GPU对内容进行计算绘制,显示到屏幕上。

我使用的demo是一个很大的CollectionView,然后为cell添加了圆角以及阴影,并且使用了大图片,所以我和这篇文章中出现的现象一致。

1、滑动列表时(即使是慢速滑动),GPU都需要计算图像、文本的动态阴影的位置和形状来进行阴影的绘制,此时GPU将成性能瓶颈,能明显观察到FPS的下降。
2、快速滑动列表时Cell每次在显示前都需要通过imageWithContentsOfFile从硬盘加载图片并解压,此时文件的IO,图片的解压让CPU也遇到性能瓶颈,使主线程无法流畅执行,让FPS雪上加霜。

原因:

CADisplayLink运行在主线程RunLoop之中,RunLoop中所管理的任务的调度时机受任务所处的RunLoopMode和CPU的繁忙程度所影响。
在第二个原因中受文件IO、解压图片的影响,RunLoop 自然无法保证CADisplayLink被调用的次数达到每秒60次,这里的调用频率正是我们的FPS指示器中所显示FPS。
而在第一个原因中主要瓶颈在于GPU,即使RunLoop能保持每秒60次调用CADisplayLink,也无法说明此时的屏幕刷新率能达到60FPS(Core Animation通过与OpenGl打交道控制GPU进行屏幕绘制),也正因为这样FPS指示器显示55+的FPS,但Instrument中的Core Animation FPS 却很低。

总结来说,我的fps一直保持在很高,只是说明runloop保持了CADisplayLink的高频率调用,但是并不能说明屏幕的刷新率也很高。这种方法在一些特殊场景下的检测并不准确,所以这种方法我不是很推荐,可以使用微信开源的matrix来进行fps的监控。

CPU使用率检测

一个进程运行的基本单位就是线程,因此一个进程中所有线程的使用率加起来就是CPU的使用率,iOS系统为我们提供了这些方法,还是需要导入#import <mach/mach.h>头文件,首先thread_basic_info为我们提供了单个线程的各种属性,其中一项就是cpu_usage

struct thread_basic_info {
    time_value_t    user_time;      /* user run time */
    time_value_t    system_time;    /* system run time */
    integer_t       cpu_usage;      /* scaled cpu usage percentage */
    policy_t        policy;         /* scheduling policy in effect */
    integer_t       run_state;      /* run state (see below) */
    integer_t       flags;          /* various flags (see below) */
    integer_t       suspend_count;  /* suspend count for thread */
    integer_t       sleep_time;     /* number of seconds that thread
                                     *  has been sleeping */
};

task_threads这个函数又为我们提供了获取当前所有线程的方法,接下来的事情就很简单了。

- (integer_t)cpuUsage {
    thread_act_array_t threads; //int 组成的数组比如 thread[1] = 5635
    mach_msg_type_number_t threadCount = 0; //mach_msg_type_number_t 是 int 类型
    const task_t thisTask = mach_task_self();
    //根据当前 task 获取所有线程
    kern_return_t kr = task_threads(thisTask, &threads, &threadCount);
    
    if (kr != KERN_SUCCESS) {
        return 0;
    }
    
    integer_t cpuUsage = 0;
    // 遍历所有线程
    for (int i = 0; i < threadCount; i++) {
        
        thread_info_data_t threadInfo;
        thread_basic_info_t threadBaseInfo;
        mach_msg_type_number_t threadInfoCount = THREAD_INFO_MAX;
        
        if (thread_info((thread_act_t)threads[i], THREAD_BASIC_INFO, (thread_info_t)threadInfo, &threadInfoCount) == KERN_SUCCESS) {
            // 获取 CPU 使用率
            threadBaseInfo = (thread_basic_info_t)threadInfo;
            if (!(threadBaseInfo->flags & TH_FLAGS_IDLE)) {
                cpuUsage += threadBaseInfo->cpu_usage;
            }
        }
    }
    assert(vm_deallocate(mach_task_self(), (vm_address_t)threads, threadCount * sizeof(thread_t)) == KERN_SUCCESS);
    return cpuUsage;
}

这样就可以获取CPU的使用率了。

个人推荐如果是想简单的进行一个性能方面的线上监控,使用这些简单的小方法就够了,如果是想对APP进行一个全量的内存监控,那么可以使用微信的Matrix。

参考文章:

APP性能监控
iOS中基于CADisplayLink的FPS指示器详解

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念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

推荐阅读更多精彩内容