上帝说要有ANR,于是Bugly就有了ANR上报。那么ANR究竟是什么?

近期非常多童鞋问起精神哥ANR的问题,那么这次就来聊一下,鸡爪怎么泡才好吃。噢不,是怎样高速定位ANR。

ANR是什么

简单说,通常就是App执行的时候,duang~卡住了。怎么搞都动不了。

当卡住超过一定时间。Android系统觉得这就是一次“ANR(Application Not Responding)”。

详细说。在下面情况发生时,会发生ANR(可能在不同ROM 中时间有所更改):

  • 用户的输入在5s内没被App响应。

  • BroadcastReceiver的onReceiver()超过10s;

  • Service中各生命周期函数运行超过20s。


ANR必须死

用户在App的绝大部分操作,都须要有App的主动回应。比方按下button之后button样式的改变、下拉滚动栏内容的移动、载入资源时的菊花转转转,它们都是“操作-反馈”配对的模式。对于我们手机上最常见的触摸操作。0.1s的响应延迟已经有非常明显的卡顿感了。而对于常见的ANR。用户至少要等5s以上!

发生了ANR,往往会弹出对话框,问用户是继续等待还是直接关掉:

相信差点儿全部Android手机用户都见过这个然并卵的ANR对话框,但大部分普通用户根本不知道这个对话框在讲什么,而且往往也仅仅有关闭App。

漫长的等待就给我看这个?从用户的体验看,就是心中一万仅仅草泥马奔腾起来撞火车的感受。可见ANR对于应用的影响并不亚于Crash。

一般来说,界面相对越不“流畅”的App(说明UI线程耗时操作多)越easy发生ANR(一个输入事件在某个设备A上4秒有了反馈。并不意味着它在其它设备B上是安全的)。

ANR事实上就是界面卡顿的极端情况。反过来,仅仅要通过合理的方案消灭了App出现的ANR,往往也同一时候会使App展示界面表现会更加顺滑流畅。

一些典型的ANR 问题场景

这里举几个easy发生ANR的场景:

1)最常见的错误,UI线程等待其他线程释放某个锁,导致UI线程无法处理用户输入。

2)游戏中每帧动画都进行了比較耗时的大量计算,导致CPU忙只是来;

3)Web应用中。网络状态不稳定。而界面在等待网络数据;

4)UI线程中进行了一些磁盘IO(包含数据库、SD卡等等)的操作。在个别设备上由于硬件损坏等原因堵塞住了;

5)手机被其它App占用着CPU。自己获取不到足够的CPU 时间片,纯属误伤。

通过ANR 日志定位问题

当ANR发生时。我们往往通过Logcat和traces文件(文件夹/data/anr/)的相关信息输出去定位问题。主要包括下面几方面:

1)基本信息,包含进程名、进程号、包名、系统build号、ANR 类型等等;

2)CPU使用信息。包含活跃进程的CPU 平均占用率、IO情况等等。

3)线程堆栈信息。所属进程包含发生ANR的进程、其父进程、近期有活动的3个进程等等。

这里举个简单的样例(实际上由于各App所处环境各异,可能出现各种各样复杂的ANR情况)当App执行卡住,弹出ANR对话框。查看Logcat输出:

ActivityManager: ANR in com.tencent.bugly.demo (com.tencent.bugly.demo/.MainActivity)
ActivityManager: PID: 18617
ActivityManager: Reason: Input dispatching timed out (Waiting because the touched window has not finished processing the input events that were previously delivered to it.)
ActivityManager: Load: 18.42 / 18.09 / 18.29
ActivityManager: CPU usage from 5924ms to 475ms ago:
ActivityManager: 93% 18617/com.tencent.bugly.demo: 93% user + 0% kernel / faults: 75 minor
……
ActivityManager: CPU usage from 2906ms to 3429ms later:
ActivityManager: 96% 18617/com.tencent.bugly.demo: 96% user + 0% kernel
……
ActivityManager: 55% TOTAL: 51% user + 3.8% kernel

分析一下,从Logcat能够得到下面信息:

  1. com.tencent.bugly.demo这个App的MainActivity发生了ANR,进程号18617;

  2. ANR原因:用户输入超时。

  3. ANR发生前、后一段时间分别附在情况:在ANR发生前后,CPU有90+%耗费在这个demo上,说明非常可能是这个demo自身性能引起的。

接下来再看traces文件确认:

----- pid 18617 at xxxx -----
Cmd line: com.tencent.bugly.demo
JNI: CheckJNI is off; workarounds are off; pins=0; globals=272 (plus 2 weak)
DALVIK THREADS:
"main" prio=5 tid=1 SUSPENDED
| group="main" sCount=1 dsCount=0 obj=0x415e4e58 self=0x415d3028
| sysTid=18617 nice=0 sched=0/0 cgrp=apps handle=1074372948
| state=S schedstat=( 38588000572 591063492 5767 ) utm=3846 stm=12 core=0
at com.tencent.bugly.demo.MainActivity$3.doCalc(MainActivity.java:~38)
at com.tencent.bugly.demo.MainActivity$3.onClick(MainActivity.java:33)
……

分析一下,traces文件里包括下面信息:

1、进程号:18617;包名:com.tencent.bugly.demo。

2、发生ANR时,main线程被挂起(也可能是其它等待状态,比方TIMED_WAIT);

3、线程的几个重要參数:

  • group:线程组名称“main”;

  • sCount:Suspended个数“1”。

  • obj:线程的Java对象地址。

  • self:线程的Native对象地址;

  • sysTid:线程号(这里主线程的线程号=进程号)“18617”。

4、详细堆栈:从堆栈能够非常清晰看出是doCalc()方法出的问题,由onClick触发。

综合以上分析。问题还原为:com.tencent.bugly.demo这个App的MainActivity中有个耗时的doCalc方法在跑。无法响应用户的触摸或按键输入。OK。接下来在代码里找问题就好了。

怎样解决ANR

当然是尽可能降低UI线程的耗时操作。以及BroadcastReceiver、Service生命周期中的标准回调方法啦。

Android官方文档建议:

1)使用AsyncTask类,能够非常方便地实现子线程耗时操作与UI更新;

2)对于BroadcastReceiver的耗时操作。建议放到Service中运行。

3)对于自建的Thread,能够通过Handler使之与UI 线程通信(这里须要注意的是。Thread默认优先级和UI线程是一样的,建议设置一般线程优先级为Process.THREAD_PRIORITY_BACKGROUND)。

这些方案大家应该都知道。只是仍难免有大量的ANR是写代码时忽略了,在測试时没发生,终于在用户的手机上出现的。回忆一下是不是都经历过用户会反馈“App卡死没反应了”。但开发GG客服MM们却又由于缺少日志或无法复现而束手无策?因此要修复ANR。首先是要能发现用户ANR了。而且能知道是哪段代码导致ANR了,这样才干谈修复。

为了帮助广大开发人员解决这一难题,腾讯Bugly针对iOS的卡顿及Android的ANR提供监測服务即将上线,协助开发人员轻松定位问题。

【腾讯bugly干货分享】精神哥手把手教你怎样智斗ANR的更多相关文章

  1. 【腾讯Bugly干货分享】微信终端跨平台组件 Mars 系列 - 我们如约而至

    导语 昨天上午,微信在广州举办了微信公开课Pro.于是,精神哥这两天的朋友圈被小龙的"八不做"刷屏了.小伙伴们可能不知道,下午,微信公开课专门开设了技术分论坛.在分论坛中,微信开源 ...

  2. 【腾讯Bugly干货分享】程序员们也该知道的事——“期权和股票”

    本文来自于腾讯Bugly公众号(weixinBugly),未经作者同意,请勿转载,原文地址:https://mp.weixin.qq.com/s/pfj9NLLuKYAfJJF84R9WAw 作者:B ...

  3. 【腾讯Bugly干货分享】聊聊苹果的Bug - iOS 10 nano_free Crash

    本文来自于腾讯Bugly公众号(weixinBugly),未经作者同意,请勿转载,原文地址:https://mp.weixin.qq.com/s/hnwj24xqrtOhcjEt_TaQ9w 作者:张 ...

  4. 【腾讯Bugly干货分享】跨平台 ListView 性能优化

    本文来自于腾讯Bugly公众号(weixinBugly),未经作者同意,请勿转载,原文地址:https://mp.weixin.qq.com/s/FbiSLPxFdGqJ00WgpJ94yw 导语 精 ...

  5. 【腾讯Bugly干货分享】让 CodeReview 这股清流再飞一会儿

    本文来自于腾讯Bugly公众号(weixinBugly),未经作者同意,请勿转载,原文地址:https://mp.weixin.qq.com/s/ToYeT4Y4pzx0ii9Z92fo-Q 作者:刘 ...

  6. 【腾讯Bugly干货分享】打造“微信小程序”组件化开发框架

    本文来自于腾讯Bugly公众号(weixinBugly),未经作者同意,请勿转载,原文地址:http://mp.weixin.qq.com/s/2nQzsuqq7Avgs8wsRizUhw 作者:Gc ...

  7. 【腾讯Bugly干货分享】总结一个技术总监的教训和经验

    本文来自于腾讯Bugly公众号(weixinBugly),未经作者同意,请勿转载,原文地址:http://mp.weixin.qq.com/s/Ry-G0Nikh6m-h3ZVC2cLyQ 导语 20 ...

  8. 【腾讯Bugly干货分享】职场中脱颖而出的成长秘诀

    本文来自于腾讯Bugly公众号(weixinBugly),未经作者同意,请勿转载,原文地址:http://mp.weixin.qq.com/s/uQKpVg7HMLfogGzzMyc9iQ 导语 时光 ...

  9. 【腾讯Bugly干货分享】OCS——史上最疯狂的iOS动态化方案

    本文来自于腾讯Bugly公众号(weixinBugly),未经作者同意,请勿转载,原文地址:https://mp.weixin.qq.com/s/zctwM2Wf8c6_sxT_0yZvXg 导语 在 ...

随机推荐

  1. iOS学习笔记11-多线程入门

    一.iOS多线程 iOS多线程开发有三种方式: NSThread NSOperation GCD iOS在每个进程启动后都会创建一个主线程,更新UI要在主线程上,所以也称为UI线程,是其他线程的父线程 ...

  2. BZOJ 3749: [POI2015]Łasuchy【动态规划】

    Description 圆桌上摆放着n份食物,围成一圈,第i份食物所含热量为c[i]. 相邻两份食物之间坐着一个人,共有n个人.每个人有两种选择,吃自己左边或者右边的食物.如果两个人选择了同一份食物, ...

  3. PHP中的验证码类(准备篇)

    <!--code.php内容--> <?php //开启session session_start(); include "vcode.class.php"; / ...

  4. Autorelease对象什么时候释放?

    Autorelease机制是iOS开发者管理对象内存的好伙伴,MRC中,调用[obj autorelease]来延迟内存的释放是一件简单自然的事,ARC下,我们甚至可以完全不知道Autorelease ...

  5. 圆桌聚餐(cogs 729)

    «问题描述:假设有来自m 个不同单位的代表参加一次国际会议.每个单位的代表数分别为ri(i=1,2,3...m), .会议餐厅共有n张餐桌,每张餐桌可容纳c i(i=1,2...n) 个代表就餐.为了 ...

  6. ElasticSearch索引自定义类型

    ES可以自动检测字段并设置映射类型.如果设置的索引类型不是我们所需要的,我们可以自行定义. Rest API设置自定义索引 首先通过ES自动映射一个IP地址的字段的类型: <pre name=& ...

  7. 从dataset表中获取某一列的所有值方法

    原文发布时间为:2008-07-31 -- 来源于本人的百度文章 [由搬家工具导入] 可以datarow遍历所有行即可,如下:pubauthor这个表中的au_lname的所有值加到listbox上面 ...

  8. 垃圾收集器与内存分配策略 (深入理解JVM二)

    1.概述 垃圾收集(Garbage Collection,GC). 当需要排查各种内存溢出.内存泄露问题时,当垃圾收集成为系统达到更高并发量的瓶颈时,我们就需要对这些“自动化”的技术实施必要的监控和调 ...

  9. [转发]Android 系统稳定性 - ANR(二)

    文章都为原创,转载请注明出处,未经允许而盗用者追究法律责任. 很久之前写的了,留着有点浪费,共享之.编写者:李文栋P.S. OpenOffice粘贴过来后格式有些混乱. http://rayleeya ...

  10. 使用Maven运行Java main的方法(转)

    使用Maven运行Java Main的方法(既Java Application项目),可以有如下方式解决: 1.将Maven项目导入到eclipse中,然后直接项目右键[Run As]->[Ja ...