阿里巴巴: 给你一个Demo 你如何快速定位ANR

Android ANR详解
本文详细介绍了Android平台上的ANR(应用程序无响应)错误,包括错误定义、常见原因、定位方式及优化策略,并提供了Traceview和Systrace等工具的具体使用方法。

本专栏专注分享大型Bat面试知识,后续会持续更新,喜欢的话麻烦点击一个关注

一、前期基础知识储备

1.ANR错误定义:在Android上,如果你的应用程序有一段时间响应不够灵敏,系统会向用户显示一个对话框,这个对话框称作“应用程序无响应”(ANR:Application Not Responding)对话框。用户可以选择“等待”而让程序继续运行,也可以选择“强制关闭”。因此,在程序里对响应性能的设计很重要,这样,系统不会显示ANR给用户。

默认情况下,在Android中Activity的最长执行时间是5秒(主要类型),BroadcastReceiver的最长执行时间的则是10秒,ServiceTimeout的最长执行时间是20秒(少数类型)。超出就会提示应用程序无响应(ANR错误)。

2.ANR错误出现原因:只有当应用程序的UI线程响应超时才会引起ANR 超时产生的原因包括:①当前事件没有机会处理,例如UI线程正在响应另外的事件,当前事件被某个事件给阻塞掉了;②当前事件正在处理 但是由于耗时太长没有能及时的完成。其他原因:③在BroadcastReceiver里做耗时的操作或计算;④CPU使用过高;⑤发生了死锁;⑥耗时操作的动画需要大量的计算工作,可能导致CPU负载过重。

二、ANR定位方式及优化

1.ANR错误定位——如果开发机器上出现ANR问题时,系统会生成一个traces.txt的文件放在/data/anr下,最新的ANR信息在最开始部分。通过adb命令将其导出到本地,输入以下字符:

$adb pull data/anr/traces.txt .

2.供选的优化ANR问题的方式:

1)为了执行一个长时间的耗时操作而创建一个工作线程最方便高效的方式是使用AsyncTask,只需要继承AsyncTask并实现doInBackground()方法来执行任务即可。为了把任务执行的进度呈现给用户,你可以执行publishProgress()方法,这个方法会触发onProgressUpdate()的回调方法。在onProgressUpdate()的回调方法中(它执行在UI线程),你可以执行通知用户进度的操作,例如:

private class DownloadFilesTask extends AsyncTask<URL, Integer, Long> {
    // Do the long-running work in here
    protected Long doInBackground(URL... urls) {
        int count = urls.length;
        long totalSize = 0;
        for (int i = 0; i < count; i++) {
            totalSize += Downloader.downloadFile(urls[i]);
            publishProgress((int) ((i / (float) count) * 100));
            // Escape early if cancel() is called
            if (isCancelled()) break;
        }
        return totalSize;
 }
 
 // This is called each time you call publishProgress()
protected void onProgressUpdate(Integer... progress) {
    setProgressPercent(progress[0]);
}

// This is called when doInBackground() is finished
protected void onPostExecute(Long result) {
    showNotification("Downloaded " + result + " bytes");
}

2)如果你实现了Thread或者HandlerThread,请确保你的UI线程不会因为等待工作线程的某个任务而去执行Thread.wait()或者Thread.sleep()。UI线程不应该去等待工作线程完成某个任务,你的UI线程应该提供一个Handler给其他工作线程,这样工作线程能够通过这个Handler在任务结束的时候通知UI线程。例如:
继承Thread类

new Thread(new Runnable() {
   @Override
   public void run() {
       /**
          耗时操作
        */
      handler.post(new Runnable() {
          @Override
          public void run() {
              /**
                更新UI
               */
          }
      });
   }
 }).start();

实现Runnable接口

class PrimeRun implements Runnable {
    long minPrime;
    PrimeRun(long minPrime) {
        this.minPrime = minPrime;
    }

    public void run() {
        // compute primes larger than minPrime
         . . .
    }
}

PrimeRun p = new PrimeRun(143);
new Thread(p).start();

使用HandlerThread

// 启动一个名为new_thread的子线程
HandlerThread thread = new HandlerThread("new_thread");
thread.start();

// 取new_thread赋值给ServiceHandler
private ServiceHandler mServiceHandler;
mServiceLooper = thread.getLooper();
mServiceHandler = new ServiceHandler(mServiceLooper);

private final class ServiceHandler extends Handler {
    public ServiceHandler(Looper looper) {
      super(looper);
    }
    
    @Override
    public void handleMessage(Message msg) {
      //默认Handler的handleMessage方法是运行在主线程中的,如果传入一个工作线程的Looper,则改变HandleMessage方法执行的所在线程
    }
}

3)开发在日常的开发过程中使用Thread或者HandlerThread,可以尝试调用Process.setThreadPriority(Process.THREAD_PRIORITY_BACKGROUND)设置较低的优先级,否则仍然会降低程序响应,因为默认Thread的优先级和主线程相同。

4)Activity的onCreate和onResume回调中尽量避免耗时的代码,应该尽可能的做比较少的事情,其实,任何执行在UI线程中的方法都应该尽可能简短快速。类似网络或者DB操作等可能长时间执行的操作,或者是类似调整bitmap大小等需要长时间计算的操作,都应该执行在工作线程中。

5)BroadcastReceiver中onReceive代码也要尽量减少耗时。如果必须在onReceive方法中执行耗时操作,建议使用IntentService进行处理,IntentService集开启线程和自动关闭服务两种功能于一身,本身非常灵活。

@Override
public void onReceive(Context context, Intent intent) {
    // This is a long-running operation
    BubbleSort.sort(data);
}
//上面的代码在onReceive方法中执行了耗时操作
@Override
public void onReceive(Context context, Intent intent) {
    // The task now runs on a worker thread.
    Intent intentService = new Intent(context, MyIntentService.class);
    context.startService(intentService);
}

public class MyIntentService extends IntentService {
   @Override
   protected void onHandleIntent(@Nullable Intent intent) {
       BubbleSort.sort(data);
   }
}
//将onReceive的耗时操作放入到IntentService中执行,执行完之后自动关闭工作线程

6)增加界面响应性(交互层面),这是一个成熟应用必备的标志—通常来说,100ms - 200ms是用户能够察觉到卡顿的上限。如果你的程序在启动阶段有一个耗时的初始化操作,可以考虑显示一个闪屏,要么尽快的显示主界面,然后马上显示一个加载的对话框,异步加载数据。无论哪种情况,你都应该显示一个进度信息,以免用户感觉程序有卡顿的情况。
##三、辅助处理ANR问题的工具
1.Traceview - 系统性能分析工具,用于定位应用代码中的耗时操作

①选好应用的进程,执行一段应用操作,从图中的上半部分,可以看到各个线程的各个方法的执行时间;

②从图中的下半部分,可以该段操作中具体调用的方法和每个方法的执行时间、执行次数。占CPU的百分比;

该图是具体的方法执行时间分布图,我们重点关注其中的“Incl Real Time”这一时间指标,其为方法的实际调用时间,单位毫秒,查看时点击Incl Real Time进行排列,方法会根据时间长短进行排列,其中超过500ms的方法我们都该重点关注。

2.Systrace - Android4.1新增的应用性能数据采样和分析工具(与google引擎联合开发 使用时借助chorme浏览器)

连接手机,进行一段操作,系统会生成一份Html文件,在谷歌浏览器中打开,如图:

①Sytrace会显示在这段操作期间所有的进程信息,在其中找到自己的进程,可以看到在测试进程中,我们定位UI Thread,可以看到里面的系统方法,这是UI渲染时的调用方法,上面有一个个的圈,绿色圈代表帧渲染时间是16.6ms(Android系统渲染UI界面时间为1秒60帧,每帧即16.6ms),超过该值的帧用红色圈标注;

②点击红色圈的标注帧,可以看到Sytrace给出的Alert,具体查看可发现,给出了该帧具体的渲染时间为36.71ms,超过16.6ms,UI渲染时会发生掉帧的情况,即卡顿,再往下看,可以看到系统给出的建议和超时的主要方法。拿到这个方法再结合Traceview工具,进行具体分析,找到自己的代码,进行优化,减少耗时。

<think>嗯,用户现在想了解如何通过指令在手机上手动触发JE(Java Exception)或ANR(Application Not Responding)错误。首先,我需要回忆一下之前关于ANR的讨论,用户之前的问题是关于删除图片文件的Linux命令,但现在转向了AndroidANR和JE触发方法。这可能是因为用户在进行Android应用测试或调试时遇到了相关问题,想要模拟这些错误来测试应用的健壮性或者调试处理机制。 首先,我需要确认用户的使用场景。用户可能是Android开发者,或者测试人员,需要手动触发这些错误来验证应用的行为。比如,当应用发生ANR时,系统会弹出对话框,用户可能想测试应用是否能正确处理这种情况,或者检查日志以优化性能。 接下来,用户的身份可能是技术相关人员,具备一定的命令行操作经验,因为问题涉及使用指令来触发错误。他们可能需要具体的步骤和命令,以及注意事项,避免对设备造成不可逆的影响。 用户的需求是明确的:通过指令手动触发JE或ANR。但深层需求可能包括了解这些错误的发生机制,如何模拟它们以进行测试,以及如何确保测试的安全性,比如不损坏设备或数据。 现在,我需要整理触发这两种错误的方法。对于JE,常见的方式是强制抛出未捕获的异常,比如在应用代码中插入崩溃点,或者通过am命令发送广播触发崩溃。而ANR通常是由于主线程被长时间阻塞,比如执行耗时操作或死循环。可以通过在Activity的UI线程中执行无限循环,或者使用adb命令模拟输入事件超时。 需要注意的是,手动触发这些错误可能会影响应用或系统稳定性,因此必须在测试环境中进行,并确保有备份。此外,某些方法可能需要root权限,或者特定的系统权限,这可能限制了部分用户的操作。 另外,用户提供的引用中提到了ANR的原因,如GC进程与ANR线程ID一致、耗时操作在主线程等,这些信息可以帮助设计触发方法。例如,在主线程执行长时间循环或大量计算可以触发ANR,而主动抛出异常可以触发JE。 最后,需要给出具体的步骤和命令,同时提醒用户注意事项,比如使用测试设备、备份数据等。可能还需要解释每种方法的原理,帮助用户理解背后的机制,以便他们根据实际情况调整。</think>#### 一、手动触发ANR的两种核心方法 (基于Android系统设计原理,主线程阻塞即触发ANR) **方法1:主线程无限循环(快速触发)** 通过adb执行Activity启动时注入死循环: ```bash adb shell am start -n com.example.app/.MainActivity --ez ANR_TEST true ``` 需在目标Activity代码中植入逻辑: ```java if (getIntent().getBooleanExtra("ANR_TEST", false)) { while (true) { // 制造主线程死循环 } } ``` **方法2:模拟输入事件超时** 通过adb持续发送无响应的触摸事件: ```bash adb shell input swipe 500 500 500 500 10000 # 10秒长滑动操作 adb shell input keyevent KEYCODE_BACK # 立即发送返回键 ``` 系统将在5秒内检测到事件未处理完成触发ANR[^2] #### 二、手动触发JE(Java Exception)的实践方案 **强制空指针异常:** ```bash adb shell am broadcast -a THROW_NPE --es target_package com.example.app ``` 需在AndroidManifest.xml声明权限: ```xml <uses-permission android:name="android.permission.DUMP" /> ``` 目标应用代码需捕获广播: ```java BroadcastReceiver crashReceiver = new BroadcastReceiver() { @Override public void onReceive(Context context, Intent intent) { throw new NullPointerException("Manual JE Trigger"); } }; ``` #### 三、高阶调试技巧 1. **ANR注入模板** 使用`am hang`命令直接冻结进程: ```bash adb shell am hang --allow-stop-package -p com.example.app ``` 系统将在60秒后强制触发ANR弹窗[^3] 2. **内存压力测试** 通过malloc反复分配大内存块: ```bash adb shell am start-activity -W -a android.intent.action.VIEW \ -d "memleak://alloc?size=1024&count=1000" com.example.app/.MainActivity ``` #### 四、操作风险控制指南 - **沙盒环境验证**:建议使用Android模拟器(如AVD)操作 - **日志实时监控**:另开终端执行`adb logcat | grep -E 'ANR|Exception'` - **系统防护措施**: ```bash adb shell settings put global development_settings_enabled 0 # 关闭开发者选项 adb shell pm disable com.example.app # 测试完成后立即冻结目标应用 ``` --相关问题--: 1. 如何通过ADB命令直接获取ANR traces.txt文件? 2. Android系统对不同类型的ANR有哪些超时阈值? 3. 如何区分JE导致的ANR和主线程阻塞导致的ANR? [^1]: ANR触发条件与主线程响应机制关联 [^2]: 输入事件分发超时检测逻辑 [^3]: `am hang`命令的系统级实现原理
评论 1
成就一亿技术人!
拼手气红包6.0元
还能输入1000个字符
 
红包 添加红包
表情包 插入表情
 条评论被折叠 查看
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值