「ANR」Android SIGQUIT(3) 信号拦截与处理

作者:非台

背景

Android的ANR频次(Application Not Responding)一直是Android用户体验的重要指标,然而在Android 6.0+的设备上,由于设备anr目录权限的收敛,已经不能通过扫描/data/anr/traces.txt文件来获取ANR文件了,因此今天我们来简单聊聊获取ANR的另一种方式,Android环境下,信号SIGQUIT(3)拦截。

信号量处理

关于信号SIGQUIT的拦截,我们需要了解信号量处理的部分相关函数,kill、signal、sigaction、sigwait、pthread_sigmask等系统信号量处理相关函数是阅读本文的必备知识,因此在这一章节简单介绍下,更多系统函数知识,请阅读《UNIX环境高级编程》。

kill [1]

头文件:#include<signal.h>

定义函数:int kill(pid_t pid,int signo)

函数说明:kill函数可以对进程发送signal,Android AMS在发生ANR的是其实是通过Process.sendSignal(pid,signal)来通信的,Process.sendSignal方法在JNI层,其实调用的是kill

想详细了解ANR的同学可以看

  • AppErrors.java:http://androidxref.com/9.0.0_r3/xref/frameworks/base/services/core/java/com/android/server/am/AppErrors.java
  • android_util_Process.cpp:http://androidxref.com/9.0.0_r3/xref/frameworks/base/core/jni/android_util_Process.cpp

signal [2]

头文件:#include<signal.h>

定义函数:sig_t signal(int signum,sig_t handler);

函数说明:signal()用于确定以后当信号sig出现时的处理方法。如果handler的值是SIG_DFL,那么就采用实现定义的缺省行为;如果handler的值是SIG_IGN,那么就忽略该信号;否则,调用handler所指向的函数(参数为信号类型)。有效的信号包括:

SIGABRT

异常终止,如调用abort()。

SIGFPE

算术运算出错,如除数为0或溢出。

SIGILL

非法函数映象,如非法指令。

SIGINT

交互式信号,如中断。

SIGSEGV

非法访问存储器,如访问不存在的内存单元。

SIGTERM

发送给本程序的终止请求信号。

signal()返回信号sig原来的handler;如果出错,则返回SIG_ERR。当随后出现信号sig时,就中断正在执行的操作,转而执行信号处理函数(*handler)(sig)。如果从信号处理程序中返回,则从中断的位置继续执行。

sigaction [3]

头文件:#include<signal.h>

定义函数:int sigaction(int signum,const struct sigaction *act ,struct sigaction *oldact)

  • 4
    点赞
  • 7
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
ANR(Application Not Responding)是Android系统中的一种错误状态,当应用程序在主线程上执行耗时操作或者阻塞UI线程超过一定时间时,系统会认为应用程序无响应,触发ANR错误。 ANR错误会导致应用程序无法响应用户的操作,用户体验变差。通常情况下,ANR错误会在以下几种情况下发生: 1. 主线程阻塞:当应用程序在主线程上执行耗时操作,例如网络请求、数据库查询等,如果这些操作耗时过长,超过了系统规定的时间限制(通常为5秒),系统就会认为应用程序无响应,触发ANR错误。 2. 主线程死锁:当应用程序中的多个线程相互等待对方释放资源而导致死锁时,主线程也会被阻塞,从而触发ANR错误。 3. 广播接收器超时:当应用程序的广播接收器在一定时间内没有处理完接收到的广播消息时,系统也会认为应用程序无响应,触发ANR错误。 为了避免ANR错误的发生,开发者可以采取以下几种措施: 1. 将耗时操作放在子线程中执行,避免阻塞主线程。可以使用AsyncTask、Thread等方式来创建子线程。 2. 使用异步操作来执行耗时操作,例如使用Handler、AsyncTask、RxJava等方式来处理耗时操作的结果。 3. 合理管理线程,避免死锁的发生。可以使用同步锁、线程池等方式来管理线程。 4. 尽量减少主线程上的工作量,例如将复杂的计算、IO操作等放在子线程中执行。 5. 对于广播接收器,尽量避免在接收到广播后执行耗时操作,可以考虑将耗时操作放在Service中执行。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值