iOS开发:监控卡顿小结

 hi~ iOS行业的小伙伴们  大家好 :

      小伙伴们:做为一个开发者,有一个学习的氛围跟一个交流圈子特别重要!!!

(小伙伴们:不忘初心 方得始终 为了生活 不要颓废 请努力拼搏!)       


      相信对于做APP的小伙伴来说,时不时会收到用户的卡顿反馈,比如:"某个用户碰到从后台切换前台卡了一下,最近偶尔会遇到几次";"某个用户反馈点一个按钮或者对话框,程序卡了好几秒";"某个用户反馈切换 tab时 很卡"等等。  

    那么界面卡顿是由哪些原因导致的呢?

     死锁:主线程拿到锁 A,需要获得锁 B,而同时某个子线程拿了锁 B,需要锁 A,这样相互等待就死锁了。

     抢锁:主线程需要访问 DB,而此时某个子线程往 DB 插入大量数据。通常抢锁的体验是偶尔卡一阵子,过会就恢复了。

      主线程大量 IO:主线程为了方便直接写入大量数据,会导致界面卡顿。

      主线程大量计算:算法不合理,导致主线程某个函数占用大量CPU。

      大量的 UI 绘制:复杂的 UI、图文混排等,带来大量的 UI 绘制。

    那怎么确定这些问题呢?


     死锁一般会伴随 crash,可以通过 crash report 来分析。

     抢锁不好办,将锁等待时间打出来用处不大,我们还需要知道是谁占了锁。

     大量 IO 可以在函数开始结束打点,将占用时间打到日志中。

     大量计算同理可以将耗时打到日志中。

     大量 UI 绘制一般是必现,还好办;如果是偶现的话,想加日志点都没地方,因为是慢在系统函数里面。

      如果可以将当时的线程堆栈捕捉下来,那么上述难题都迎刃而解。主线程在什么函数哪一行卡住,在等什么锁,而这个锁又是被哪个子线程的哪个函数占用,有了堆栈,就可以知道。是慢在UI绘制,还是慢在我们的代码。

    解决方案


     在移动设备上开发软件,性能一直是我们最为关心的话题之一,我们作为程序员除了需要努力提高代码质量之外,及时发现和监控软件中那些造成性能低下的”罪魁祸首”也是我们神圣的职责。

    众所周知,iOS平台因为UIKit本身的特性,需要将所有的UI操作都放在主线程执行,所以也造成不少程序员都习惯将一些线程安全性不确定的逻辑,以及其它线程结束后的汇总工作等等放到了主线,所以主线程中包含的这些大量计算、IO、绘制都有可能造成卡顿。

     在Xcode中已经集成了非常方便的调试工具Instruments,它可以帮助我们在开发测试阶段分析软件运行的性能消耗,但一款软件经过测试流程和实验室分析肯定是不够的,在正式环境中由大量用户在使用过程中监控、分析到的数据更能解决一些隐藏的问题。

    寻找卡顿的切入点监控卡顿,最直接就是找到主线程都在干些啥玩意儿.我们知道一个线程的消息事件处理都是依赖于NSRunLoop来驱动,所以要知道线程正在调用什么方法,就需要从NSRunLoop来入手.CFRunLoop的代码是开源,其中核心方法CFRunLoopRun简化后的主要逻辑大概是这样的:


   不难发现NSRunLoop调用方法主要就是在kCFRunLoopBeforeSources和kCFRunLoopBeforeWaiting之间,还有kCFRunLoopAfterWaiting之后,也就是如果我们发现这两个时间内耗时太长,那么就可以判定出此时主线程卡顿.

   量化卡顿的程度

   要监控NSRunLoop的状态,我们需要使用到CFRunLoopObserverRef,通过它可以实时获得这些状态值的变化,具体的使用如下:


    只需要另外再开启一个线程,实时计算这两个状态区域之间的耗时是否到达某个阀值,便能揪出这些性能杀手.

    为了让计算更精确,需要让子线程更及时的获知主线程NSRunLoop状态变化,所以dispatch_semaphore_t是个不错的选择,另外卡顿需要覆盖到多次连续小卡顿和单次长时间卡顿两种情景,所以判定条件也需要做适当优化.将上面两个方法添加计算的逻辑如下:


  记录卡顿的函数调用

    监控到了卡顿现场,当然下一步便是记录此时的函数调用信息,此处可以使用一个第三方Crash收集组件PLCrashReporter,它不仅可以收集Crash信息也可用于实时获取各线程的调用堆栈,使用示例如下:

PLCrashReporterConfig config = [[PLCrashReporterConfig alloc] initWithSignalHandlerType:PLCrashReporterSignalHandlerTypeBSD

symbolicationStrategy:PLCrashReporterSymbolicationStrategyAll];

PLCrashReporter 

crashReporter = [[PLCrashReporter alloc] initWithConfiguration:config];

NSData data = [crashReporter generateLiveReport];

PLCrashReport 

reporter = [[PLCrashReport alloc] initWithData:data error:NULL];

NSString *report = [PLCrashReportTextFormatter stringValueForCrashReport:reporter

withTextFormat:PLCrashReportTextFormatiOS];

NSLog(@"------------\n%@\n------------", report);

     当检测到卡顿时,抓取堆栈信息,然后在客户端做一些过滤处理,便可以上报到服务器,通过收集一定量的卡顿数据后经过分析便能准确定位需要优化的逻辑,至此这个实时卡顿监控就大功告成了!


   小伙伴们:其实做为一个开发者,有一个学习的氛围跟一个交流圈子特别重要!!!


扩大圈子,技术交流,学习提升!

↓ (机会永远是留给那些有准备的人)  ↓       

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

推荐阅读更多精彩内容

  • 1.ios高性能编程 (1).内层 最小的内层平均值和峰值(2).耗电量 高效的算法和数据结构(3).初始化时...
    欧辰_OSR阅读 29,891评论 8 265
  • OC语言基础 1.类与对象 类方法 OC的类方法只有2种:静态方法和实例方法两种 在OC中,只要方法声明在@int...
    奇异果好补阅读 9,804评论 0 11
  • 前言 在移动设备上开发软件,性能一直是我们最为关心的话题之一,我们作为程序员除了需要努力提高代码质量之外,及时发现...
    路飞_Luck阅读 20,415评论 12 88
  • 在南方待的那些时日里,是巨怀念北方浓烈的太阳的。 虽然三月末的北方,寒意依旧不减于二月,但有阳光的日子,总是让人心...
    哀而不伤的忆阅读 3,110评论 0 2
  • 点击题目下方唯美感情学,一键关注本账号 唯美感情学每个女人的内心都有一段不想提及的故事,每个女人都需要一个内心倾述...
    唯美感情学阅读 1,495评论 0 0