Activity出现ANR以及如何避免?

什么是ANR

ANR:Application Not Responding

ANR定义:在Android上,如果你的应用程序有一段时间响应不够灵敏,系统会向用户展示一个对话框,这个对话框称作应用程序无响应(ANR : Application Not Responding)对话框.用户可以选择"等待"而让程序继续运行,也可以选择强制关闭.所以一个流畅合理的应用程序中不能出现ANR,而让用户每次都要处理这个对话框.因此,在程序里对响应性能的设计很重要,这样系统不会显示ANR给用户.默认情况下,在Android中Activity的最长执行时间是5秒,BroadcastReceiver的最长执行时间则是10秒.

什么会引发ANR?

在Android里,应用程序的响应性是由Activity Manager和WindowManager系统服务监视的.当它检测到以下情况的一个时,Android就会针对特定的应用程序显示ANR:

  • 1.在5秒内没有相应输入的事件(例如,按键按下,屏幕触摸)
  • 2.BroadcastReceiver在10秒内没有执行完毕

造成以上两点的原因有很多,比如在主线程中做了非常耗时的操作,比如下载,io异常等.
潜在的耗时操作,例如网络或数据库操作,或者高耗时的计算(如改变位图尺寸),应该在子线程里(或者以数据库操作为例,通过异步请求的方式)来完成.然而,不是说你的主线程阻塞在哪里等待子线程的完成—也不是调用Thread.sleep()或者Thread.wait().替代的方法是,主线程应该为子线程提供一个handler,以便完成时能够提交给主线程.

如何避免ANR?
  • 1.运行在主线程里的任何方法都尽可能少的做事情.特别是,Activity应该在它的关键生命周期方法(如onCreate()和onResume())里尽可能少的去做创建操作.(可以采用重新开启子线程的方式,然后使用Handler+Message的方式做一些操作,比如更新主线程中的UI等)
  • 2.应用程序应该避免在BroadcastReceiver里做耗时的操作或计算.但不再是在子线程里做这些任务(因为BroadcastReceiver的生命周期短),替代的是,如果响应Intent广播需要执行一个耗时的动作的话,应用程序应该启动一个Service(需要注意的是可以在广播接受者中启动Service,但是却不可以在Service中启动BroadcastReceiver).
  • 3.避免在Intent Receiver里启动一个Activity,因为它会创建一个新的画面,并从当前用户正在运行的程序上抢夺焦点.如果你的应用程序在响应Intent广播时需要向用户展示什么,你应该使用Notification Manager来实现.

总结:主要的解决办法就是不要再主线程中做耗时的操作,而应放在子线程中来实现,比如采用Handler+message的方式,或者是有时候需要做一些和网络相互交互的耗时操作就采用asynTask异步任务的方式(它的底层其实是Handler+message有所区别的是它是线程次)等,在主线程中更新UI.

ActivityManagerService是负责监控应用程序的响应情况的服务,当应用程序出现ANR时,该服务会弹出ANR对话框。其ANR相关的源码如下: 1. 监控应用程序响应状态:ActivityManagerService会周期性地检测所有应用程序的响应状态,如果应用程序长时间未响应,就会触发ANR,代码如下: ``` private void checkAppTimeLimitLocked(ProcessRecord app) { // Check if the app has exceeded its time limits. if (app.notResponding) { return; // Don't kill the app if it is already not responding. } final boolean isSilentANR = mService.mDidDexOpt; final long now = SystemClock.uptimeMillis(); final long maxTime = isSilentANR ? mConstants.SANE_ANR_TIMEOUT : mConstants.ANR_TIMEOUT; final long timeSinceUnfrozen = now - app.lastStateTime; final boolean overTimeLimit = timeSinceUnfrozen > maxTime; if (DEBUG_ANR) { Slog.i(TAG, "checkAppTimeLimitLocked: " + app + " timeSinceUnfrozen=" + timeSinceUnfrozen + " maxTime=" + maxTime + " over=" + overTimeLimit); } if (overTimeLimit) { makeAppNotRespondingLocked(app, null, null, 0, "time limit exceeded", isSilentANR); } } ``` 2. 触发ANR:当应用程序长时间未响应时,ActivityManagerService会调用makeAppNotRespondingLocked方法触发ANR,代码如下: ``` void makeAppNotRespondingLocked(ProcessRecord app, String reason, String shortMsg, int flags, String message, boolean isSilentANR) { if (app.notResponding) { return; } mService.addErrorToDropBox("anr", app, app.processName, null, null, reason, message, null, null); ... // Show the ANR to the user. Message msg = Message.obtain(); HashMap<String, Object> map = new HashMap<String, Object>(); map.put("app", app); map.put("shortMsg", shortMsg); map.put("flags", flags); map.put("isSilentANR", isSilentANR); msg.obj = map; mHandler.sendMessage(msg); ... } ``` 3. 弹出ANR对话框:当应用程序出现ANR时,ActivityManagerService会通过Handler向主线程发送消息,弹出ANR对话框,代码如下: ``` private final class UiHandler extends Handler { public UiHandler() { super(mService.mHandler.getLooper()); } @Override public void handleMessage(Message msg) { switch (msg.what) { case SHOW_NOT_RESPONDING_UI_MSG: { HashMap<String, Object> data = (HashMap<String, Object>) msg.obj; ProcessRecord app = (ProcessRecord) data.get("app"); String shortMsg = (String) data.get("shortMsg"); int flags = (Integer) data.get("flags"); boolean isSilentANR = (Boolean) data.get("isSilentANR"); ... // Display the ANR dialog. mActivityManagerInternal.showAppErrorUi(app.pid, app.info.packageName, shortMsg, null, null, flags, isSilentANR); ... break; } ... } } } ``` 以上就是ActivityManagerService中ANR相关的源码。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值