Android---App 崩溃

news2024/11/24 4:58:18

崩溃问题是衡量 App 质量的决定性考核标准。Android 系统会输出各种相应的 log 日志,很大程度上降低了工程师 debug 崩溃问题的难度。如果要给 crash 日志进行分类,可以分为2大类:JVM 异常(Exception)堆栈信息native 代码崩溃日志

JVM 异常堆栈信息

Java 中异常(Exception)分两种:检查异常(checked Exception)和非检查异常(unchecked Exception)

\bullet 检查异常是在代码编译时期,Android Studio 就会提示代码有错误,无法通过编译。比如 IOException。如果没有在代码中将这些异常 catch,而是直接抛出,最终也有可能导致程序崩溃。

\bullet 非检查异常包括 error 和运行时异常(RuntimeException)。AS 并不会在编译时期提示这些异常信息,而是在程序运行时期因为代码错误而直接导致程序崩溃。比如 OOM 或者空指针异常(NPE)

Java 异常

对于上述两种异常,我们都可以使用 UncaughtExceptionHandler 来进行捕获操作。它是 Thread 的一个内部接口,定义如下

对于传入的 Thread,如果因为未捕获异常而导致被终止,uncaughtException 则会被调用。我们可以通过它来间接捕获程序异常,并进行异常信息的记录工作,或者给出更友好的异常提示信息。 

自定义异常处理类

自定义类实现 UncaughtExceptionHandler 接口,并实现 uncaughtException() 方法。如下代码所示

需要注意的几点,自定义异常处理类中,需要持有线程默认异常处理类,如图中红框处所示。这样做的目的是在自定义异常处理类无法处理或者处理异常失败时,还可以将异常交给系统做默认处理。如果自定义异常处理类成功处理异常,需要进行页面跳转或者将程序进程杀死,否则程序会一直卡死在崩溃界面,并弹出无响应对话框。在上面代码中通过 CrashDisplayActivity 页面来展示所捕获的异常详细信息。

一般情况下,在 handleException() 方法中需要做以下几件事情:

1)收集 crash 的现场相关信息。如下面方法获取当前 App 的版本信息,以及所有手机设备的相关信息

2)日志的记录工作,将收集到的信息保存在本地。比如以文件的方式保存。从下图中可以看出,除了我们自己收集的日志,还需要将系统抛出的异常信息也保持到文件中,方便后续开发人员分析问题原因。

使用自定义异常处理类

上面我们提到的 LagouExceptionHandler 自定义异常处理类定义好之后,就可以将其初始化。并将主线程注册到 LagouExceptionHandler 中。如下

最终,报错的 crash 日志信息格式如下图所示

需要注意的是,因为使用了文件写操作,所以需要动态申请文件操作的权限。

native 异常

当程序中的 native 代码发生崩溃时,系统会在 /data/tombstones/ 目录下保存一份详细的崩溃日志信息。如果一个 native crash 是必现的,不妨在模拟器上重现 bug。并将 /data/tombstones 中的崩溃日志拉到本地电脑中加以分析。

比如,创建一个模拟 native crash 的项目 LagouNativeCrash。项目结构如下

MainActivity 中有一个点击按钮 Button,当点击此按钮时,会调用 crash() 方法触发 native 代码崩溃。

native crash 在 native-lib.cpp 文件中声明,如下

当点击 Button 触发 native 崩溃之后,系统会在 /data/tombstones 目录下生成 tombstones 日志文件。可以在此日志文件中查看详细的保存信息。如下所示

但是,如果 native crash 是偶发现象,并且在模拟器上一时难以复现。那么就需要将工作交给测试工程师,在真机上尝试复现。

这种情况下就需要一种机制,将 native crash 现场的日志信息保存到可以访问的手机目录中。目前比较成熟,使用也比较广泛的是谷歌的 Breakpad。Breadpad 是一个跨平台的开源库,也可以在其 Breakpad Github 上下载自己编译,并通过 JNI 的方式引入到项目中。

NDK 导入 Breakpad

在 Breakpad GitHub 官网上,有一个 README Android 的介绍文件,这个文件专门介绍了如何在 Android 项目中导入 Breakpad。我们可以直接使用 cmake 方式将其编译为一个静态库。

在捕获 native crash 之前,需要初始化 Breakpad。主要是设置 Breakpad 保存 crash 日志的路径,如下所示

图中传入的 path 就是 Breakpad 保存日志的文件目录,一般情况下保存在外置 SDK 目录。

初始化好之后,就可以在我们自己的 native 业务层模拟一个崩溃现场。Breakpad 会自动捕获这次 crash,并将生成的 crash 信息保存在所设置的目录中。

breadpad 生成的文件是 .dmp 文件,需要将其进行转换,如下所示

线上崩溃日志捕获

上面介绍了 java 和 native 崩溃的捕获都是基于现场能够复现 bug 的前提。但是,对于线上用户,这种操作方式是不太现实的。首先,不可能将一个 debug 版本的 APK 安装到终端用户的手机上;其次,不太可能要求用户操作一遍手机并将某一目录下的文件发送给开发人员。

对于大多数公式来说,针对线上版本没有必要自己实现一个抓取 log 的平台系统。最快速的实现方式就是集成第三方 SDK。目前比较成熟,采用也比较多的就是腾讯的 Bugly。Bugly 能够满足线上版本捕获 crash 的所有需求,包括 java 层和 native 层的 crash 都可以获取相应的日志。并且每天 Bugly 都会邮件通知上一天的崩溃日志,方便测试和开发统计 bug 的分布以及崩溃率。

其它的 crash 上报工具,比如 XCrashSentry

这两者比 Bugly 好的地方就是除了自动拦截界面崩溃事件,还可以主动上报错误信息。以 XCrash 为例,基本使用如下所示

可以看出 XCrash 的使用更加灵活,工程师的掌控性更高。可以通过设置不同的过滤方式针对性的上报相应的 crash 日志。并且在捕获到 crash 之后可以加入自定义的操作,比如本地保存日志或者直接进行网络上传。

Sentry 还有一个好处就是可以通过设置过滤,来判断是否上报 crash 日志。这对于 SDK 的开发人员是很有用的,比如一些 SDK 的开发商只是想收集自身 SDK 引入的 crash,对于用户的其他操作导致的 crash 进行过滤。这种情况就可以考虑集成 Sentry。

总结

本次主要介绍了 Android 崩溃的相关知识。对于 Android 工程师来说,crash 可以分为2类:Java 层和 Native 层

\bullet Java 层一般通过自定义 UncaughtExceptionHandler 进行异常拦截

\bullet Native 层可以考虑集成谷歌的 breakpad 进行捕获,并保存日志在本地

最后介绍了几个线上捕获 crash 的工具:Bugly、XCrash 和 Sentry。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/1194085.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

爆肝一文,走进大名鼎鼎的HTTP协议(通俗白话+三万字超详细+抓包工具使用)

文章目录 前言1. HTTP 是什么1.1 HTTP 完整请求流程1.2 理解 HTTP 协议的工作过程 2. HTTP 协议格式2.1 抓包工具的使用2.2 抓包工具的原理2.3 抓包结果2.4 协议格式总结 3. HTTP 请求(Request)3.1 认识 URL(Uniform Resource Locator)URL 基本格式关于 URL encode 3.2 认识请求…

2023.11.10联赛 T4题解

题目大意 题目思路 我们考虑分块处理。 我们可以维护一个状态,表示块内每个字母对应的真实字母,因为只有 3 3 3个字母,所以只有 6 6 6种情况。 对于每一个块,我们可以对于每种状态、每种块,预处理出以 A A A或 B B …

商城免费搭建之java商城 java电子商务Spring Cloud+Spring Boot+mybatis+MQ+VR全景+b2b2c

1. 涉及平台 平台管理、商家端(PC端、手机端)、买家平台(H5/公众号、小程序、APP端(IOS/Android)、微服务平台(业务服务) 2. 核心架构 Spring Cloud、Spring Boot、Mybatis、Redis 3. 前端框架…

19. 深度学习 - 用函数解决问题

文章目录 Hi, 你好。我是茶桁。 上一节课,我们从一个波士顿房价的预测开始写代码,写到了KNN。 之前咱们机器学习课程中有讲到KNN这个算法,分析过其优点和缺点,说起来,KNN这种方法比较低效,在数…

【前端】TypeScript核心知识点讲解

1.TypeScript简介及入门案例 (1)什么是TypeScript? TypeScript 是 JavaScript 的一个超集,支持 ECMAScript 6 (ES6)标准。 TypeScript 由微软开发的自由和开源的编程语言。 TypeScript 设计目标是开发大…

ubuntu 16.04.5 安装 vivado 2019.1 完整编译AD9361的环境

一、前期安装 1、安装ncurses库(已经包含了,其他的os需要安装) sudo apt install libncurses5二、安装 sudo ./xsetup使用lic进行激活。 三、安装后 输入指令 sudo gedit ~/.bashrc 末尾添加 source /opt/Xilinx/Vivado/2019.1/setti…

串口通信(11)-CRC校验介绍算法

本文为博主 日月同辉,与我共生,csdn原创首发。希望看完后能对你有所帮助,不足之处请指正!一起交流学习,共同进步! > 发布人:日月同辉,与我共生_单片机-CSDN博客 > 欢迎你为独创博主日月同…

条件渲染-if/hidden

1.条件渲染的方法 1.1 wx:if方法(少用) block只是一个容器,容器内的都用于if判断 1.2 hidden方法 注:view要小写,否则不生效 1.3方法对比

【狂神说Java】Dubbo + Zookeeper

✅作者简介:CSDN内容合伙人、信息安全专业在校大学生🏆 🔥系列专栏 :狂神说Java 📃新人博主 :欢迎点赞收藏关注,会回访! 💬舞台再大,你不上台,永远…

【java:牛客每日三十题总结-4】

java:牛客每日三十题总结 总结如下 总结如下 集合相关知识点 元素是否排序和插入顺序无关,取决与集合实现是否考虑了传入对象的java.lang.Comparable接口抽象类和接口相关知识 只能说越来越抽象了 java线程通信的方式 在Java中,常用的线程通信方式有两…

轻松下载网页音频和视频

在网页上看到好看的视频或者听到的音乐想保存,让我来教你(仅供学习) 注意:有极少部分的网站视频经过加密,无法下载 一、视频下载 1.打开视频网页 2.右键“检查” 3.刷新网页 4.按照下图中步骤操作 5.把复制的链接放…

【漏洞复现】BYTEVALUE智能流控路由器存在命令执行

【漏洞介绍】 百为智能流控路由器 /goform/webRead/open 路由的 ?path 参数存在有回显的命令注入漏洞。攻击者可通过该漏洞在服务器端执行命令,写入后门,获取服务器权限,从而获取路由器权限。 【指纹】 title”BYTEVALUE 智能流控路由器”…

C语言C位出道心法(五):内存管理

C语言C位出道心法(一):基础语法 C语言C位出道心法(二):结构体|结构体指针|链表 C语言C位出道心法(三):共用体|枚举 C语言C位出道心法(四):文件操作 C语言C位出道心法(五):内存管理 一:C语言内存管理认知 二:C语言中内存堆|栈认知 三:C语言中引用内存丢失认知

数字档案馆实施方案需要考虑哪些因素

数字档案馆实施方案通常需要考虑以下几个方面: 1. 硬件和软件设备的选择:需要根据数字档案馆的规模和需求选购适当的硬件和软件设备,包括服务器、存储设备、数据库、操作系统、专久智能档案管理系统等。 2. 数据的采集和处理:确定…

idea中搭建Spring boot项目(借助Spring Initializer)

创建新项目 启动端口 在项目配置文件application.properties中写入 #启动端口server.port8088编写测试方法 创建控制类文件夹–>便于规范我们新建一个controller包–>建一个HelloWorld.class package com.example.hellospringboot.controller;import org.springframew…

【工具推荐】一键多平台文章发布神器推荐(免费)

hello,大家好,我是你们老朋友洛林,上一篇文章说到自己深受多平台手动发布的折磨「传送门」,准备开发一款文章多平台工具,后来联系到 Wechatsync 原作者进行了简单的沟通,下面是关于以后的一些规划&#xff…

MySQL 批量修改表的列名为小写

1、获取脚本 SELECT concat( alter table , TABLE_NAME, change column , COLUMN_NAME, , lower( COLUMN_NAME ), , COLUMN_TYPE, comment \, COLUMN_COMMENT, \; ) AS 脚本 FROM information_schema.COLUMNS WHERE TABLE_SCHEMA 数据库名 and TABLE_NAME表名-- 大写是up…

让你认识C++中的模板

目录 一. 泛型编程1、定义 二、函数模板1、定义2、格式3、函数模板的实例化(1)、强制转化(2)、显式实例化 三、类模板1、 类模板的定义格式2、实例3、 类模板的实例化 一. 泛型编程 1、定义 泛型编程:编写与类型无关…

Spring封装数据结果

Spring封装数据结果 POST请求JSON格式 基本数据类型 public class Demo {private byte aByte;private short aShort;private int anInt;private long aLong;private float aFloat;private double aDouble;private char aChar;private boolean aBoolean; }没有传键 封装时就会…

arm2 day4

汇编编写流水灯 代码: LED流水灯现象: