App崩溃异常捕获与处理

原因说明:由于我们写的代码难免会出现一些bug,以及由于测试环境和生产环境差异等导致在测试过程中没有发现问题,而在app上线之后会偶然出现的一些bug,如app在使用过程中出现ANR、app卡死、黑屏等现象。
解决思路:我们开发者需要去捕获这些异常,收集这些异常信息,并且上传到服务器,利于开发人员去解决这些问题,同时,我们还需要给用户一个友好的交互体验

(1) 自定义一个应用异常捕获类 CrashHandler 实现Thread.UncaughtExceptionHandler接口,并重写uncaughtException方法 来 按我们自己的方式处理异常

  • 单例模式获取捕获类实例
  • 初始化捕获类
  • 重写uncaughtException方法处理异常
package comi.example.liy.firstbasicproject.tool;

import android.content.Context;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.os.Build;
import android.os.Looper;
import android.util.Log;
import android.widget.Toast;

import java.io.File;
import java.io.PrintWriter;
import java.lang.reflect.Field;
import java.text.SimpleDateFormat;
import java.util.Date;

import static comi.example.liy.firstbasicproject.tool.GlobalConstant.localLogDirectory;

/**
 * Created by liy on 2017-06-22.
 * 全局异常捕获的机制
 * (1)自定义异常收集类CrashHandler并实现UncaughtExceptionHandler接口,并保留系统默认异常处理。
 * (2)然后实现uncaughtException方法,当程序发生Uncaught异常的时候,有该方法来接管程序(并可记录或发送错误报告到服务器)
 */

public class CrashHandler implements Thread.UncaughtExceptionHandler {
    public static final String TAG = CrashHandler.class.getName();

    private Context mContext;//程序的Context对象
    private File logFile = null;//错误记录文件
    private Thread.UncaughtExceptionHandler mDefaultHandler;//系统默认的UncaughtException处理类

    /** 
     * 保证只有一个CrashHandler实例 
     */
    private static CrashHandler instance;//CrashHandler实例
    
    private CrashHandler() {

    }
    //获取CrashHandler实例:在整个app中全局捕获异常,所以我们自定义的捕获类是单例模式
    public static CrashHandler getInstance() {
        if (instance == null)
            instance = new CrashHandler();
        return instance;
    }

    /**
     * 初始化捕获
     */
    public void init(Context context) {
        mContext = context;
        logFile = new File(localLogDirectory);
        mDefaultHandler = Thread.getDefaultUncaughtExceptionHandler();//获取系统默认的UncaughtException处理器
        Thread.setDefaultUncaughtExceptionHandler(this);//设置该CrashHandler为程序的默认异常处理器
    }

    /**
     * 当UncaughtException发生时会转入该函数来处理。在线程因未捕获的异常而临近死亡时被调用。
     */
    @Override
    public void uncaughtException(Thread thread, Throwable ex) {
        // 如果我们没处理异常,并且系统默认的异常处理器不为空,则交给系统默认的异常处理器来处理
        if (!handleException(ex) && mDefaultHandler != null) {
            mDefaultHandler.uncaughtException(thread, ex);  // 系统处理
        } else {
            /*upLoadErrorFileToServer(logFile);//已经记录完log,也可以把这个记录的错误日志发到服务端*/
            try {
                Thread.sleep(3000);
              } catch (InterruptedException e) {
                Log.e(TAG, "error : ", e);
              }

            //捕获异常马上关闭app,即将app整个进程杀死
            android.os.Process.killProcess(android.os.Process.myPid());// 退出程序
        }

    }

    /**
     * 自定义异常处理:收集错误信息至本地(可选)、发送错误报告到服务器(可以将整个文件上传 或者 上传异常信息的字符串)等
     * true代表处理该异常,不再向上抛异常,
     * false代表不处理该异常(可以将该log信息存储起来)然后交给上层(这里就到了系统的异常处理)去处理
     */
    private boolean handleException(final Throwable ex) {
        if (ex == null) {
            return false;
        }

        //提示对话框:使用Toast来显示异常信息
        new Thread() {
            @Override
            public void run() {
                Looper.prepare();
                Log.v("exception_info","Exception Info:" + ex + "");
                Toast.makeText(mContext,"Exception Info:"+ex.getMessage(), Toast.LENGTH_LONG).show();
                Looper.loop();
            }
        }.start();

        //保存到sd卡:记录异常信息,把错误信息记录到手机卡中,如果异常为空,则把权限交给系统。
        PrintWriter pw = null;
        try {
            File parentDir = new File(logFile.getParent());
            if(!parentDir.exists()){
                parentDir.mkdirs();
            }
            if(!logFile.exists()){
                try{
                    logFile.createNewFile();
                }catch (Exception e){
                    Log.v("exception_mkdirs","exception info-->>"+e.getMessage());
                }
            }
            pw = new PrintWriter(logFile);
            //收集记录错误信息至本地
            collectInfoToSDCard(pw, ex);
            pw.close();
        } catch (Exception e) {
            e.printStackTrace();
        }
        return true;
    }

    /**
     * 收集记录错误信息至本地
     **/
    private void collectInfoToSDCard(PrintWriter pw, Throwable ex) throws PackageManager.NameNotFoundException, IllegalAccessException, IllegalArgumentException {
        PackageManager pm = mContext.getPackageManager();
        PackageInfo mPackageInfo = pm.getPackageInfo(mContext.getPackageName(), PackageManager.GET_ACTIVITIES);

        pw.println("time: " + new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date())); // 记录错误发生的时间
        pw.println("versionCode: " + mPackageInfo.versionCode); // 版本号
        pw.println("versionName: " + mPackageInfo.versionName); // 版本名称

        //Build类,这个类定义了所有关于手机的一些参数,如版本号,系统名称,Android版本等。然后通过反射机制,把这些信息和错误信息一起记录到日志里面。
        Field[] fields = Build.class.getDeclaredFields();//获取所有属性
        for (Field field : fields) {
            field.setAccessible(true);//可以读取private属性并可对其进行更改
            pw.print(field.getName() + " : ");
            pw.println(field.get(null).toString());
        }
        ex.printStackTrace(pw);
    }

}

保存在sd卡中的异常文件格式(示例):参考

保存在sd卡中的异常文件格式.png

(2) 全局捕获异常:自定义Application,并在AndroidManifest.xml中进行配置

public class APPAplication extends Application {
    @Override
    public void onCreate() {
        // TODO Auto-generated method stub
        super.onCreate();

        // 捕捉异常:初始化自定义的异常捕获类
        CrashHandler crashHandler = CrashHandler.getInstance();
        crashHandler.init(getApplicationContext());
    }
}
<application
        android:allowBackup="true"
        android:icon="@drawable/jackboard"
        android:label="@string/app_name"
        android:supportsRtl="true"
        android:theme="@style/AppTheme1"
        android:name=".APPAplication">
</application>

(3) activity中捕获异常

  • HomeActivity
/**
 * Created by liy on 2018-05-30.
 */

public class HomeActivity extends Activity  {
    private Button btnCrashHandler;
    private CrashHandler crashHandler;

    @Override
    public void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_home);
        initViews();
        initData();
        initListeners();
    }

    @Override
    public void initViews() {
        btnCrashHandler = (Button)findViewById(R.id.activity_main_crashHandler);
    }

    @Override
    public void initData() {
        crashHandler = CrashHandler.getInstance();
        crashHandler.init(this);
    }

    @Override
    public void initListeners() {
        btnCrashHandler.setOnClickListener(new View.OnClickListener() {
            @Override
            public void onClick(View v) {
                Integer a = null;
                a.toString();//模拟一个空指针异常,触发nullpointer运行时错误
            }
        });
    }

    @Override
    protected void onDestroy() {
        super.onDestroy();
    }
    
}
  • ActivityGeneralInterface接口
public interface ActivityGeneralInterface {

    void initViews();
    void initData();
    void initListeners();
}
  • activity_home.xml
<?xml version="1.0" encoding="utf-8"?>
<LinearLayout xmlns:android="http://schemas.android.com/apk/res/android"
    android:orientation="horizontal"
    android:layout_width="match_parent"
    android:layout_height="match_parent">
    <Button
          android:id="@+id/activity_main_crashHandler"
          android:layout_width="match_parent"
          android:layout_height="wrap_content"
          android:text="CrashHandler"
          android:textAllCaps="false"/>
</LinearLayout>
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,163评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,301评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,089评论 0 352
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,093评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,110评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,079评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,005评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,840评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,278评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,497评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,667评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,394评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,980评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,628评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,796评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,649评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,548评论 2 352

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,061评论 25 707
  • 大家都知道,现在安装Android系统的手机版本和设备千差万别,在模拟器和自己的android手机上运行良好的程序...
    Orz013阅读 3,705评论 0 4
  • 【Android Activity】 什么是 Activity? 四大组件之一,通常一个用户交互界面对应一个 ac...
    Rtia阅读 3,800评论 3 18
  • 面试必背 会舍弃、总结概括——根据我这些年面试和看面试题搜集过来的知识点汇总而来 建议根据我的写的面试应对思路中的...
    luoyangzk阅读 6,753评论 6 173
  • 房价涨疯了的时候才意识到自己过去的几年没有狠下心买房是这辈子犯的最大的错误之一。 我知道我已经错过...
    范范_嘟嘟阅读 135评论 0 0