检测方式
线上环境的ANR错误需要我们及时的上报服务器,那么如何主动检测ANR错误呢?
1、FileObserver
当发生ANR的时候,我们查看控制台的日志
07-27 10:25:09.732 23995-24002/com.qunar.yuzhiyun.anr I/dalvikvm: threadid=3: reacting to signal 3
07-27 10:25:09.792 23995-24002/com.qunar.yuzhiyun.anr I/dalvikvm: Wrote stack traces to '/data/anr/traces.txt'
可以看出,发生ANR,进程会收到信号3,紧接着开始向’/data/anr/traces.txt’ 文件写 栈跟踪信息,这样的话,我们只需要通过FileObserver 来监控’/data/anr/traces.txt’ 文件的变化不就行了吗?只要有变化就说明发生了ANR(据笔者所知,暂时没有其他引起traces.txt变化的原因)。但是FileObserver在5.0及以上系统部分机型失效,所以不算好的方法。
2、watchDog方式
有人提出watchDog方式的检测方法,在主线程定义一个变量count,在子线程不断的通知主线程去更新count的值(比如+=1),子线程维护一个值与count相等的变量,睡眠5秒后去判断两个值是否相等,从而判断是否出现了ANR错误,以下代码简单的给出了示范。(读者在运行这段代码的时候,先点击FloatingActionButton 让主线程长时间睡眠,然后再操作一下界面的其他元素,就可以在控制台看到Log.e(“Thread”,”ANR happpened”); 打印的日志啦)
public class MainActivity extends AppCompatActivity {
Handler handler=new Handler();
int count;
private final Runnable message= new Runnable() {
@Override public void run() {
count= (count+ 1) % 10;
}
};
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
Toolbar toolbar = (Toolbar) findViewById(R.id.toolbar);
setSupportActionBar(toolbar);
new Thread(){
@Override
public void run() {
int lastCount;
while(true) {
lastCount = count;
handler.post(message);
try {
Thread.sleep(5000);
} catch (InterruptedException e) {
e.printStackTrace();
}
if(count == lastCount) {
//检测到了ANR
Log.e("Thread","ANR happpened");
}
}
}
}.start();
FloatingActionButton fab = (FloatingActionButton) findViewById(R.id.fab);
fab.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
try {
Log.i("Thread","start sleep");
Thread.sleep(1000000000);
} catch (InterruptedException e) {
e.printStackTrace();
}
}
});
}
}
参考
http://mobnav.dev.qunar.com/2016/08/29/anr_monitor_analysis/