anr发生原因
Application Not Responding(简称:ANR)指应用中一些特定的事件(如用户触摸事件、广播等)在应用的主线程没有在规定的时间内处理完,系统自动做出终止应用运行的响应。Android系统中,ActivityManagerService(简称AMS)和WindowManagerService(简称WMS)会检测App的响应时间,如果App在特定时间无法相应屏幕触摸或键盘输入时间,或者特定事件没有处理完毕,就会出现ANR。
anr出现的场景:
InputDispatching Timeout:5秒内无法响应屏幕触摸事件或键盘输入事件。
BroadcastQueue Timeout :在执行前台广播(BroadcastReceiver)的onReceive()函数时10秒没有处理完成,后台为60秒。
Service Timeout :前台服务20秒内,后台服务在200秒内没有执行完毕。
ContentProvider Timeout :ContentProvider的publish在10s内没进行完。
1、查看是否有anr文件
在命令行窗口
adb shell
ls adb /data/anr
2、导出anr日志
法1:
在android studio terminal窗口
adb bugreport
法2:
打开CMD小黑框,进入到Android SDK 目录下的platform-tools文件夹下面直接输出 : adb logcat 复制粘贴到一个txt文本下,方便稍后查看
法3:
执行adb pull /data/anr/traces.txt D:\traces.txt 输出traces日志。
anr日志分析
例1
try {
Thread.sleep(9000);
} catch (InterruptedException e) {
e.printStackTrace();
}
在activity中按钮点击后在主线程执行sleep操作。导出的anr日志如下:
搜索DALVIK_THREADS,找到第一个,然后看主线程main,tid是线程id。然后线程状态是Sleeping。继续往下看堆栈可以看到是因为按钮点击后sleep时间过长,导致后续点击事件无法及时处理导致。
各参数含义:
group:线程所处的线程组
sCount: 线程被正常挂起的次数
dsCount: 线程因调试而挂起次数
nice:线程的调度有优先级
utm:线程在用户态中调度时间值
stm:线程在内核态中的调度时间值
core:最后执行这个线程的CPU核序号
例2
public class MainActivity extends AppCompatActivity implements View.OnClickListener {
private static final String TAG = "MainActivity";
private Button button;
private Button button2;
private Button btnMVTest;
private Object object= new Object();
private MyLock myLock = new MyLock();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
findViews();
}
private void findViews() {
button = findViewById(R.id.button);
button2 = findViewById(R.id.button2);
btnMVTest = findViewById(R.id.btnMVTest);
button.setOnClickListener(this);
button2.setOnClickListener(this);
btnMVTest.setOnClickListener(this);
}
@Override
public void onClick(View v) {
int viewId = v.getId();
switch (viewId){
case R.id.button2:
testBlock();
break;
case R.id.btnMVTest:
Log.e(TAG, "onClick: mainThread="+Thread.currentThread().getName()+" tid="+Thread.currentThread().getId());
doSqlQuery();
break;
}
}
private void testBlock(){
Thread thread = new Thread(new Runnable() {
@Override
public void run() {
synchronized (myLock){
try {
doSqlQuery();
Thread.sleep(11000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
}
});
thread.setName("DTestBlock");
thread.start();
}
private void doSqlQuery(){
synchronized (myLock){
Log.e(TAG, "doThreadBlcok: 被锁住的函数 threadName="+Thread.currentThread().getName()+" threadId="+Thread.currentThread().getId());
}
}
private class MyLock{
}
}
看下anr日志。
可以看到主线程处于Blocked(阻塞)状态,为何阻塞:
- waiting to lock <0x0b19381d> (a com.example.router.MainActivity$MyLock) held by thread 3
尝试锁定MyLock实例对象,但是它被线程3所持有。
线程3执行了sleep,在sleep的时间段内,没有释放对象锁。导致主线程获取不到对象锁而处于阻塞状态,后续点击事件响应不了,从而导致anr。
3、总结
anr是比较严重的性能问题,对用户体验影响较大。而且手动导出anr日志有时并没那么方便,可以考虑使用第三方性能检测工具如bugly等,简化anr的排查。