Android ANR发生的原因总结和解决办法

本文介绍了Android平台上的ANR(Application Not Responding)现象,包括其定义、触发条件、常见原因及预防措施。ANR通常发生在主线程长时间无响应时,如处理耗时任务或与其他进程竞争资源。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

ANR定义

ANR的全称是application not responding,是指应用程序未响应,Android系统对于一些事件需要在一定的时间范围内完成,如果超过预定时间能未能得到有效响应或者响应时间过长,都会造成ANR。一般地,这时往往会弹出一个提示框,告知用户当前xxx未响应,用户可选择继续等待或者Force Close。

ANR发生条件

首先ANR的发生是有条件限制的,分为以下三点:

1.只有主线程才会产生ANR,主线程就是UI线程;
2.必须发生某些输入事件或特定操作,比如按键或触屏等输入事件,在BroadcastReceiver或Service的各个生命周期调用函数;
3.上述事件响应超时,不同的context规定的上限时间不同

  • a.主线程对输入事件5秒内没有处理完毕
  • b.主线程在执行BroadcastReceiver的onReceive()函数时10秒内没有处理完毕
  • c.主线程在Service的各个生命周期函数时20秒内没有处理完毕。
ANR发生原因

那么导致ANR的根本原因是什么呢?简单的总结有以下两点:

1.主线程执行了耗时操作,比如数据库操作或网络编程
2.其他进程(就是其他程序)占用CPU导致本进程得不到CPU时间片,比如其他进程的频繁读写操作可能会导致这个问题。
细分的话,导致ANR的原因有如下几点:

1.耗时的网络访问
2.大量的数据读写
3.数据库操作
4.硬件操作(比如camera)
5.调用thread的join()方法、sleep()方法、wait()方法或者等待线程锁的时候
6.service binder的数量达到上限
7.system server中发生WatchDog ANR
8.service忙导致超时无响应
9.其他线程持有锁,导致主线程等待超时
10.其它线程终止或崩溃导致主线程一直等待

如何避免ANR发生

那么如何避免ANR的发生呢或者说ANR的解决办法是什么呢?

1.避免在主线程执行耗时操作,所有耗时操作应新开一个子线程完成,然后再在主线程更新UI。
2.BroadcastReceiver要执行耗时操作时应启动一个service,将耗时操作交给service来完成。
3.避免在Intent Receiver里启动一个Activity,因为它会创建一个新的画面,并从当前用户正在运行的程序上抢夺焦点。如果你的应用程序在响应Intent广 播时需要向用户展示什么,你应该使用Notification Manager来实现。

### Android ANR 解决方案与处理流程 #### 1. ANR 的定义及产生原因 ANR 是指应用程序未响应,通常是因为主线程被阻塞导致无法及时响应用户输入或消息广播。Android 系统对一些操作有严格的时间限制,例如输入事件处理、广播接收等,如果在规定时间内没有得到处理,就会触发 ANR[^1]。 #### 2. 常见的 ANR 类型 - **输入事件分发超时**:当主线程在 5 秒内未能完成输入事件的处理时,会触发 ANR。 - **广播处理超时**:当主线程在 10 秒内未能完成广播接收器的任务时,会触发 ANR。 - **服务启动超时**:当主线程在 20 秒内未能完成服务的启动任务时,会触发 ANR[^2]。 #### 3. ANR 的解决方法 ##### 优化主线程逻辑 将耗时任务移至后台线程是解决 ANR 的关键。可以通过以下方式实现: - **AsyncTask**:适用于简单的异步任务,但需要注意其生命周期问题。 - **HandlerThread**:提供一个带有 Looper 的线程,适合需要频繁通信的任务。 - **WorkManager**:用于执行延迟或周期性任务,特别适合需要保证任务执行的任务[^2]。 ##### 使用调试工具分析 ANR trace.txt 文件是 Android 系统在发生 ANR 时自动生成的日志文件,记录了系统中所有线程的堆栈信息,特别是主线程的状态。通过分析这个文件,可以定位导致 ANR 的根本原因[^3]。 ###### 获取 trace.txt 文件 从设备中提取使用 ADB 命令获取 ANR 堆栈: ```bash adb pull /data/anr/traces.txt ``` 在 logcat 中标记文件生成路径每次发生 ANR 时,logcat 会记录 trace.txt 的路径。在调试工具中直接查看部分集成开发环境(如 Android Studio)可以在运行时直接查看 ANR 信息。 ###### 分析 trace.txt 的结构 trace.txt 文件包含所有线程的堆栈信息,开发者可以通过查找主线程的状态来确定阻塞的原因。例如,如果主线程处于 `wait` 或 `sleep` 状态,可能是因为某个耗时操作未被移至后台线程[^3]。 #### 4. 避免 ANR 的最佳实践 - **减少主线程负担**:确保主线程只处理 UI 更新轻量级任务,避免执行耗时操作。 - **合理使用线程池**:通过线程池管理后台任务,避免创建过多线程导致资源耗尽。 - **监控应用性能**:使用工具如 Systrace StrictMode 监控应用性能,及时发现潜在问题[^1]。 #### 5. 示例代码 以下是一个将耗时任务移至后台线程的示例: ```java new Thread(() -> { // 耗时操作 try { Thread.sleep(5000); // 模拟耗时任务 } catch (InterruptedException e) { e.printStackTrace(); } runOnUiThread(() -> { // 更新 UI textView.setText("Task Completed"); }); }).start(); ``` #### 6. 总结 通过优化主线程逻辑、使用调试工具分析 ANR 并遵循最佳实践,开发者可以有效避免解决 ANR 问题,提升应用性能用户体验[^2]。 ###
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值