Android 面试题 ANR 五
🔥 什么是 ANR 🔥
ANR
(Application Not Responding )
应用无响应的简称,是为了在APP
卡死时,用户 可以强制退出APP
的选择,从而避免卡机无响应问题,这是Android
系统的一种自我保护机制。
在
Android
中,应用程序响应由Activity Manager
和Window Manager
系统服务进行监视。ANR(Application Not Responding )
,则是Android
的一种自我保护措施,当主线程出现卡顿时候,Android
系统会给用户一个弹出提示,让用户手动选择继续等待还是强制关闭此APP
。
🔥 ANR 类型 🔥
按键响应分发超时(Key Dispatch Timeout) 默认 5 s
,超过则会出现ANR。广播超时(Broadcast Timeout) 默认 10 s
,超过则会出现ANR。服务超时(Service Timeout) 默认 20 s
,超过则会出现ANR。
KeyDispatchTimeout(常见)
input事件在5S内没有处理完成发生了ANR。
logcat日志关键字:Input event dispatching timed out
BroadcastTimeout
前台Broadcast:onReceiver在10S内没有处理完成发生ANR。
后台Broadcast:onReceiver在60s内没有处理完成发生ANR。
logcat日志关键字:Timeout of broadcast BroadcastRecord
ServiceTimeout
前台Service:onCreate,onStart,onBind等生命周期在20s内没有处理完成发生ANR。
后台Service:onCreate,onStart,onBind等生命周期在200s内没有处理完成发生ANR
logcat日志关键字:Timeout executing service
ContentProviderTimeout
ContentProvider 在10S内没有处理完成发生ANR。
logcat日志关键字:timeout publishing content providers
🔥 产生原因 🔥
在
Android
系统中,APP
通常运行在一个UI Thread
或者叫MainThread
里。并且Android
中只有一个MainThread
和Main Message Queue
。MainThread
主要用于UI
的绘制、事件响应,监听与接收事件处理等功能。Main Message Queue
主要存放用户要处理消息的队列,主线程MainThread
从消息队列Main Message Queue
中取消息Message
后,尽快分发下去,一旦某条消息分发超时,则ANR
可能发生。
1、主线程进行耗时操作:如数据库读写、IO读写、网络请求等
2、多线程操作的死锁,主线程被block;
3、主线程被Binder 对端block;
4、System Server中WatchDog出现ANR;
5、service binder的连接达到上线无法和和System Server通信
6、系统资源已耗尽(管道、CPU、IO)
🔥 从以下几个疑问点进行分析 🔥
1、为什么
APP
不能获取CPU
时间片?
2、APP
是否是等待一些没能及时处理的事件完成?3、消息处理流程是不是太复杂?
🔥 常见解决方案 🔥
1.在主线程里尽可能的少做事,将耗时操作都放到子线程中执行。特别是在Acitivty生命周期(onCreate/onResume())里减少创建操作,潜在的耗时操作。(网络请求、数据库操作、计算位图尺寸等)放到子线程,然后通过Handler提交到主线程,而不是wait()sleep()等在那
2.别再广播里做耗时操作。放到Service里或者子线程。
3.如果真有耗时操作的话,可以做一个进度条,避免用户以为死机了。
🔥 分析解决 ANR问题 🔥
在分析
ANR
时有一些常见的模式可供选择:
APP
正在主线程上进行缓慢的I/O
操作。
APP
正在主线程中进行很复杂的计算操作主线程正在对另一个进程执行同步
Binder
程序调用,但另一个进程需要很长时间才能返回结果。主线程在等待另一个正在长时间执行块操作的子线程时被阻塞。
主线程因为另一个线程死锁,无论是
Bind
调用还是主线程调用,都不能让主线程等待很久,更不能在主线程中进行复杂的计算。知道产生ANR的原因,那么如何避免ANR 问题呢?
Strict mode
使用
StrictMode
可以帮助您在开发应用程序时在主线程上发现意外的I/O
操作。 您可以在application
或activity
使用StrictMode
。
关闭 ANR Dialog 提示
查看方法ANR控制的方法:设置---- 开发者选项---
显示所有ANR
注意 :
如没有开发者选项,请进入设置---关于手机--- 多次连击 版本号 即可打开隐藏的开发者选项
Traceview
Traceview获取正在运行的应用程序的跟踪信息,分析此
traces.txt
文件 可以推测出主线程在忙于某些事情。
traces
文件通常保存在/data/anr/traces.txt
下,你可以直接用adb cat
查看,或者adb pull
出来都可以。adb root adb remount adb pull /data/anr/traces.txt .
![]()
🔥 ANR 问题分析解决建议 🔥
1. 耗时操作
请放在工作线程中进行,可以使用
Handler、AsyncTask
等。
2. IO 操作
(比如:网络操作、存储操作等)也是引起ANR的常见因素。强烈建议在工作线程中进行。
3. 程序锁竞争
某些情况,
ANR
产生的原因不是直接因为在主线程中产生的。 比如: 工作线程对某个资源
等上锁,恰好此时,主线程需要此资源
,如等待超时,则此时ANR可能发生。
4. 死锁
当主线程因为请求一个其他线程正在持有的资源而进入等待状态时,
ANR
可能会发生。
5. 广播接收慢
应用程序可以通过广播接收器响应广播消息,例如启用或禁用飞行模式或更改连接状态。 当应用程序花费太长时间来处理广播消息时,理论上超过10s 未处理完成,
ANR
可能会发生。
6.广播 ANR发生在下列情况下:
onReceive()
方法长时间未执行完毕。尽量避免在onReceive()
中进行耗时操作。onReceive 方法中进行耗时操作,超过10s 未处理,会引起ANR
广播接收者调用
goAsync()
方法并且未能在PendingResult
对象上调用finish()
。您的广播接收机可以使用
goAsync()
来通知系统需要更多的时间来处理消息。 但是,您应该在PendingResult
对象上调用finish()
。 以下示例显示如何调用finish()以让系统回收广播接收器并避免ANR:
如要处理的广播内容较多,请使用
IntentService
进行处理。
🔥 捕获与监控 🔥
在开发阶段出现ANR
在开发阶段出现还是比较好解决的,只需要看报错的日志就可以了。然后定位到具体的代码行数。没找到的话,可以找
/data/anr/traces.tet
文件,这个文件记录了ANR日志。
/data/anr/traces.tet
文件:
主要看1.CPU、用户态、内核态、IO的利用率。哪个利用率高了,就是那个出问题了。
2.堆栈的利用率。
3.看“main” 的报错情况,自己写的错误一般都在这。
线上部分手机出现ANR
1.集成第三方:bugly、蒲公英、极光等监控,在那看ANR发生的手机型号、错误日志等。然后去找同款手机去测试一下(没有可以去云测)
2.自己监控: FileObserver、watchdog、FileObserver
FileObserver是一个抽象类,定义一个子类去继承他,实现onEvent方法。然后监控/data/anr/这个目录下有没有新文件产生,有的话就说明有ANR,然后传到服务器上。
watchdog : android系统就用的这个
🔥 Crash监控 🔥
Crash也就是闪退。闪退分为Java层的闪退和native层的闪退。
Java层的Crash
闪退就是发生了没有捕获的异常,处理不了了,然后就弹一个框“程序崩溃”,然后app就退出了。
1.获取UncaughtExceptionHandler Thread.getDefaultUncaughtExceptionHandler();
2.实现UncaughtExceptionHandler接口
3.在这个方法里就有了uncaughtException
在这里可以将报错信息Throwable存到本地,然后发送给服务器(时间、手机型号、app版本等自己写)
然后最重要的还得调用系统的uncaughtException方法,不然程序虽然不会崩溃,但是会卡死、无限重启。
系统怎么处理的
RuntimeInit.java -> KillApplicationHandler
也是实现UncaughtExceptionHandler接口
两个步骤 :
在try里通过AMS弹窗
在finally里杀死app进程
相关文章:

Android 面试题 ANR 五
🔥 什么是 ANR 🔥 ANR(Application Not Responding )应用无响应的简称,是为了在 APP卡死时,用户 可以强制退出APP的选择,从而避免卡机无响应问题,这是Android系统的一种自我保护机制。 在Android中…...
实训笔记7.28
实训笔记7.28 7.28笔记一、Hive的基本使用1.1 Hive的命令行客户端的使用1.2 Hive的JDBC客户端的使用1.2.1 使用前提1.2.2 启动hiveserver21.2.3 使用方式 1.3 Hive的客户端中也支持操作HDFS和Linux本地文件 二、Hive中DDL语法2.1 数据库的管理2.1.1 创建语法2.1.2 修改语法2.1.…...
C 游游的二进制树
题目描述 游游拿到了一棵树,共有nnn个节点,每个节点都有一个权值:0或者1。这样,每条路径就代表了一个二进制数。 游游想知道,有多少条路径代表的二进制数在[l,r][l,r][l,r]区间范围内? (请注意…...
收发存和进销存有什么区别?
一、什么是收发存和进销存 1、收发存 收发存是供应链管理中的关键概念,用于描述企业在供应链中的物流和库存管理过程。 收发存代表了企业在采购、生产和销售过程中的物流活动和库存水平。 收(Receiving) 企业接收供应商送达的物料或产品…...
小程序 账号的体验版正式版的账号信息及相关配置
siteinfo.js // 正式环境 const releaseConfig {appID: "",apiUrl: "",imgUrl: "" }; // 测试环境(包含开发环境和体验环境) const developConfig {appID: "",apiUrl: "",imgUrl: "" }…...
AIGC(Artificial Intelligence Generated Content)和 Web3对比,未来发展
一、AIGC(Artificial Intelligence Generated Content)行业 历史背景 AIGC(Artificial Intelligence Generated Content)是指利用人工智能技术生成的内容。随着人工智能技术的不断发展,AIGC 行业逐渐兴起。早期的 AIG…...

机器学习之Boosting和AdaBoost
1 Boosting和AdaBoost介绍 1.1 集成学习 集成学习 (Ensemble Learning) 算法的基本思想就是将多个分类器组合,从而实现一个预测效果更好的集成分类器。 集成学习通过建立几个模型来解决单一预测问题。它的工作原理是生成多个分类器/模型,各自独立地学…...
汇编语言预定义寄存器和协处理器
ARM汇编器对ARM的寄存器和协处理器进行了预定义(包括APCS对r0~r15寄存器的定义),所有的寄存器和协处理器名都是大小写敏感的。 (1)预定义寄存器名 下面列出了被ARM汇编器预定义的寄存器名。 r0ÿ…...

【前缀和】974. 和可被 K 整除的子数组
Halo,这里是Ppeua。平时主要更新C,数据结构算法,Linux与ROS…感兴趣就关注我bua! 974. 和可被 K 整除的子数组 题目:示例:题解: 题目: 示例: 题解: 本题与560.和为K的子数组高度相似 同样的,本题利用了前缀和的定理.当(pre[i]-…...

linux页框回收之shrink_node函数源码剖析
概述 《Linux内存回收入口_nginux的博客-CSDN博客》前文我们概略的描述了几种内存回收入口,我们知道几种回收入口最终都会调用进入shrink_node函数,本文将以Linux 5.9源码来描述shrink_node函数的源码实现。 函数调用流程图 scan_control数据结构 str…...

网络运维基础问题及解答
前言 本篇文章是对于网络运维基础技能的一些常见问题的解答,希望能够为进行期末复习或者对网络运维感兴趣的同学或专业人员提供一定的帮助。 问题及解答 1. 列举 3 种常用字符编码,简述怎样在 str 和 bytes 之间进行编码和解码。 答:常用的…...
【RabbitMQ】之保证数据不丢失方案
目录 一、数据丢失场景二、数据可靠性方案 1、生产者丢失消息解决方案2、MQ 队列丢失消息解决方案3、消费者丢失消息解决方案 一、数据丢失场景 MQ 消息数据完整的链路为:从 Producer 发送消息到 RabbitMQ 服务器中,再由 Broker 服务的 Exchange 根据…...

插入排序算法
插入排序 算法说明与代码实现: 以下是使用Go语言实现的插入排序算法示例代码: package mainimport "fmt"func insertionSort(arr []int) {n : len(arr)for i : 1; i < n; i {key : arr[i]j : i - 1for j > 0 && arr[j] > …...

Linux标准库API
目录 1.字符串函数 2.数据转换函数 3.格式化输入输出函数 4.权限控制函数 5.IO函数 6.进程控制函数 7.文件和目录函数 1.字符串函数 2.数据转换函数 3.格式化输入输出函数 #include<stdarg.h>void test(const char * format , ...){va_list ap;va_start(ap,format…...

腾讯云—自动挂载云盘
腾讯云,稍微麻烦了点。 腾讯云服务器,镜像为opencloudos 8。 ### 1、挂载云盘bash #首先通过以下命令,能够看到新的数据盘,如果不能需要通过腾讯云控制台卸载后,重新挂载,并重启服务器。 fdisk -l#为 /dev…...

为Win12做准备?微软Win11 23H2将集成AI助手:GPT4免费用
微软日前确认今年4季度推出Win11 23H2,这是Win11第二个年度更新。 Win11 23H2具体有哪些功能升级,现在还不好说,但它会集成微软的Copilot,它很容易让人想到多年前的“曲别针”助手,但这次是AI技术加持的,Co…...

Opencv Win10+Qt+Cmake 开发环境搭建
文章目录 一.Opencv安装二.Qt搭建opencv开发环境 一.Opencv安装 官网下载Opencv安装包 双击下载的软件进行解压 3. 系统环境变量添加 二.Qt搭建opencv开发环境 创建一个新的Qt项目(Non-Qt Project) 打开创建好的项目中的CMakeLists.txt,添加如下代码 # openc…...
Matlab实现光伏仿真(附上30个完整仿真源码)
光伏发电电池模型是描述光伏电池在不同条件下产生电能的数学模型。该模型可以用于预测光伏电池的输出功率,并为优化光伏电池系统设计和控制提供基础。本文将介绍如何使用Matlab实现光伏发电电池模型。 文章目录 1、光伏发电电池模型2、使用Matlab实现光伏发电电池模…...
JSON.stringify()与JSON.parse()
JSON.parse() 方法用来解析 JSON 字符串 onst json {"result":true, "count":42}; const obj JSON.parse(json); console.log(typeof(json)) //string console.log(typeof(obj)) //objJSON.stringify() 方法将一个 JavaScript 对象或值转换为 JSON 字…...

neo4j教程-安装部署
neo4j教程-安装部署 Neo4j的关键概念和特点 •Neo4j是一个开源的NoSQL图形存储数据库,可为应用程序提供支持ACID的后端。Neo4j的开发始于2003年,自2007年转变为开源图形数据库模型。程序员使用的是路由器和关系的灵活网络结构,而不是静态表…...
基于服务器使用 apt 安装、配置 Nginx
🧾 一、查看可安装的 Nginx 版本 首先,你可以运行以下命令查看可用版本: apt-cache madison nginx-core输出示例: nginx-core | 1.18.0-6ubuntu14.6 | http://archive.ubuntu.com/ubuntu focal-updates/main amd64 Packages ng…...
大语言模型如何处理长文本?常用文本分割技术详解
为什么需要文本分割? 引言:为什么需要文本分割?一、基础文本分割方法1. 按段落分割(Paragraph Splitting)2. 按句子分割(Sentence Splitting)二、高级文本分割策略3. 重叠分割(Sliding Window)4. 递归分割(Recursive Splitting)三、生产级工具推荐5. 使用LangChain的…...

Springcloud:Eureka 高可用集群搭建实战(服务注册与发现的底层原理与避坑指南)
引言:为什么 Eureka 依然是存量系统的核心? 尽管 Nacos 等新注册中心崛起,但金融、电力等保守行业仍有大量系统运行在 Eureka 上。理解其高可用设计与自我保护机制,是保障分布式系统稳定的必修课。本文将手把手带你搭建生产级 Eur…...
linux 下常用变更-8
1、删除普通用户 查询用户初始UID和GIDls -l /home/ ###家目录中查看UID cat /etc/group ###此文件查看GID删除用户1.编辑文件 /etc/passwd 找到对应的行,YW343:x:0:0::/home/YW343:/bin/bash 2.将标红的位置修改为用户对应初始UID和GID: YW3…...
vue3 定时器-定义全局方法 vue+ts
1.创建ts文件 路径:src/utils/timer.ts 完整代码: import { onUnmounted } from vuetype TimerCallback (...args: any[]) > voidexport function useGlobalTimer() {const timers: Map<number, NodeJS.Timeout> new Map()// 创建定时器con…...

多模态大语言模型arxiv论文略读(108)
CROME: Cross-Modal Adapters for Efficient Multimodal LLM ➡️ 论文标题:CROME: Cross-Modal Adapters for Efficient Multimodal LLM ➡️ 论文作者:Sayna Ebrahimi, Sercan O. Arik, Tejas Nama, Tomas Pfister ➡️ 研究机构: Google Cloud AI Re…...
Pinocchio 库详解及其在足式机器人上的应用
Pinocchio 库详解及其在足式机器人上的应用 Pinocchio (Pinocchio is not only a nose) 是一个开源的 C 库,专门用于快速计算机器人模型的正向运动学、逆向运动学、雅可比矩阵、动力学和动力学导数。它主要关注效率和准确性,并提供了一个通用的框架&…...
C#中的CLR属性、依赖属性与附加属性
CLR属性的主要特征 封装性: 隐藏字段的实现细节 提供对字段的受控访问 访问控制: 可单独设置get/set访问器的可见性 可创建只读或只写属性 计算属性: 可以在getter中执行计算逻辑 不需要直接对应一个字段 验证逻辑: 可以…...

云原生安全实战:API网关Kong的鉴权与限流详解
🔥「炎码工坊」技术弹药已装填! 点击关注 → 解锁工业级干货【工具实测|项目避坑|源码燃烧指南】 一、基础概念 1. API网关(API Gateway) API网关是微服务架构中的核心组件,负责统一管理所有API的流量入口。它像一座…...

[免费]微信小程序问卷调查系统(SpringBoot后端+Vue管理端)【论文+源码+SQL脚本】
大家好,我是java1234_小锋老师,看到一个不错的微信小程序问卷调查系统(SpringBoot后端Vue管理端)【论文源码SQL脚本】,分享下哈。 项目视频演示 【免费】微信小程序问卷调查系统(SpringBoot后端Vue管理端) Java毕业设计_哔哩哔哩_bilibili 项…...