【FastDev4Android框架開發】Android崩潰異常捕捉CustomCrash,提升用戶體驗(五)

轉載請標明出處:
http://www.lxweimin.com/p/70a32892d134
本文出自:【江清清博客-代號獨狼】
(一):寫在前面的話
接著上一篇繼續更新,上一篇文章已經把FastDev4Android項目數據輕量級緩存ACache組件做了講解和使用。今天項目更新客戶端崩潰異常捕捉組件CustomCrash的講解和使用。
本文章中的例子代碼已經同步到FastDev4Android項目中,地址為:
https://github.com/jiangqqlmj/FastDev4Android
在平時我們都知道,Android系統的手機和設備千差萬別,在模擬器上運行良好的程序安裝到某款手機上說不定就出現崩潰的現象,設備比較多,所以在程序發布出去之后,如果出現了崩潰現象,開發者應該及時獲取在該設備上導致崩潰的信息,同時可以獲取設備的相關信息和用戶信息,這對于下一個版本的 BUG 修復幫助極大,所以今天就來介紹一下如何在程序崩潰的情況下收集相關的設備參數信息和具體的異常信息,并發送這些信息到服務器供開發者分析和調試程序。同時替換系統默認的崩潰彈框,提升應用的用戶體驗。
首先我們來看一下系統默認的崩潰彈框顯示:

1.png

看上面的運行狀態,一旦我們的應用出現了異常崩潰,立馬會彈框,點擊OK應用就退出了,這樣用戶也不知道發生了什么情況,一下子應用的用戶體驗下降了很多。更加嚴重的是,作為我們開發者還不知道云在用戶手機的APP什么時候崩潰的,到底因為什么原因崩潰的。 OK下面我們來具體實現自定義的攔截崩潰異常的功能;
(二):具體實現
2.1:定義類 首先我們需要自定義一個實現Thead.UncaughtExceptionHandler的類,然后實現內部接口中定義的方法: void uncaughtException(Thread thread, Throwable ex); 具體如下:

public class CustomCrash  implements Thread.UncaughtExceptionHandler
/*
     * (non-Javadoc) 進行重寫捕捉異常
     *
     * @see
     * java.lang.Thread.UncaughtExceptionHandler#uncaughtException(java.lang
     * .Thread, java.lang.Throwable)
     */
    @Override
    public void uncaughtException(Thread thread, Throwable ex) {
        if(type_save==TYPE_SAVE_SDCARD){
            // 1,保存信息到sdcard中
            saveToSdcard(mContext, ex);
        }else if(type_save==TYPE_SAVE_REMOTE){
            // 2,異常崩潰信息投遞到服務器
            saveToServer(mContext,ex);
        }
        // 3,應用準備退出
        showToast(mContext, "很抱歉,程序發生異常,即將推出.");
        try {
            Thread.sleep(3500);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        ManagerActivity.getInstance().finishActivity();
        android.os.Process.killProcess(android.os.Process.myPid());

    }

2.2.上下文注冊 我們需要把當前應用的上下文注冊到系統的異常處理器中。這樣就讓系統執行我們自定義的異常捕捉器。

public void setCustomCrashInfo(Context pContext) {
        this.mContext = pContext;
        Thread.setDefaultUncaughtExceptionHandler(this);
    }

2.3:崩潰異常日志保存
①:數據保存到SDCard中,直接轉換異常日志信息,寫入SDCard文件中如下:

/**
     * 保存異常信息到sdcard中
     *
     * @param pContext
     * @param ex
     *            異常信息對象
     */
    private void saveToSdcard(Context pContext, Throwable ex) {
        String fileName = null;
        StringBuffer sBuffer = new StringBuffer();
        // 添加異常信息
        sBuffer.append(getExceptionInfo(ex));
        if (Environment.getExternalStorageState().equals(
                Environment.MEDIA_MOUNTED)) {
            File file1 = new File(CRASH_SAVE_SDPATH);
            if (!file1.exists()) {
                file1.mkdir();
            }
            fileName = file1.toString() + File.separator + paserTime(System.currentTimeMillis()) + ".log";
            File file2 = new File(fileName);
            FileOutputStream fos;
            try {
                fos = new FileOutputStream(file2);
                fos.write(sBuffer.toString().getBytes());
                fos.flush();
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }

②:數據POST投遞到服務器中,進行接口存入相應的文件或者數據庫中

 /**
     * 進行把數據投遞至服務器
     * @param pContext
     * @param ex  崩潰異常
     */
    private void saveToServer(Context pContext,Throwable ex){
        final String carsh_log=getExceptionInfo(ex);
        new Thread(new Runnable() {
            @Override
            public void run() {
                HashMap<String,String> params=new HashMap<String,String>();
                params.put("crash_log",carsh_log);
                String result= IoUtils.responseFromServiceByGetNo(CARSH_LOG_DELIVER, params);
                if (result.equals("1")){
                    Log.d(TAG,"崩潰日志投遞成功...");
                }else {
                    Log.d(TAG,"崩潰日志投遞失敗...");
                }
            }
        }).start();
    }

2.4:異常捕捉器具體使用: 我們需要在自定義的Application初始化方法進行初始化我們的捕捉器,然后設置改變系統捕捉器處理方式:

@Override
    public void onCreate() {
        super.onCreate();
        this.instance=this;
        //初始化崩潰日志收集器
        CustomCrash mCustomCrash=CustomCrash.getInstance();
        mCustomCrash.setCustomCrashInfo(this);

    }

OK下面我們來具體來使用一下,我們故意制造一個空指針異常:結果分別如下:


2.png

3.png

4.png

5.png

6.png

2.5:由于上面對于類的核心方法進行了講解,并且該類其他也沒有多少行代碼,方便大家閱讀這邊直接把整個類復制如下,并且為了大家測試使用,類中崩潰日志投遞的地址也是可以正常使用的。

package com.chinaztt.fda.crash;

import android.content.Context;
import android.os.Environment;
import android.os.Looper;
import android.widget.Toast;

import com.chinaztt.fda.utils.IoUtils;
import com.chinaztt.fda.utils.Log;
import com.chinaztt.fda.utils.ManagerActivity;
import com.chinaztt.fda.utils.StrUtils;

import java.io.File;
import java.io.FileOutputStream;
import java.io.PrintWriter;
import java.io.StringWriter;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.HashMap;
import java.util.TimeZone;

/**
 * 當前類注釋:客戶端運行 異常崩潰數據撲捉異常保存SD卡或者實時投遞服務器工具類
 * 項目名:FastDev4Android
 * 包名:com.chinaztt.fda.crash
 * 作者:江清清 on 15/10/26 13:29
 * 郵箱:jiangqqlmj@163.com
 * QQ: 781931404
 * 公司:江蘇中天科技軟件技術有限公司
 */
public class CustomCrash  implements Thread.UncaughtExceptionHandler{
    private static final String TAG="CustomCrash";
    private static final int TYPE_SAVE_SDCARD=1; //崩潰日志保存本地SDCard  --建議開發模式使用
    private static final int TYPE_SAVE_REMOTE=2; //崩潰日志保存遠端服務器 --建議生產模式使用

    private int type_save=2;  //崩潰保存日志模式 默認為2,采用保存Web服務器
    private static final String CRASH_SAVE_SDPATH="sdcard/fda_cache/"; //崩潰日志SD卡保存路徑
    private static final String CARSH_LOG_DELIVER="http://img2.xxh.cc:8080/SalesWebTest/CrashDeliver";
    private static CustomCrash instance = new CustomCrash();
    private Context mContext;
    private CustomCrash() {
    }
    /**
     *
     * @return
     */
    public static CustomCrash getInstance() {
        return instance;
    }

    /*
     * (non-Javadoc) 進行重寫捕捉異常
     *
     * @see
     * java.lang.Thread.UncaughtExceptionHandler#uncaughtException(java.lang
     * .Thread, java.lang.Throwable)
     */
    @Override
    public void uncaughtException(Thread thread, Throwable ex) {
        if(type_save==TYPE_SAVE_SDCARD){
            // 1,保存信息到sdcard中
            saveToSdcard(mContext, ex);
        }else if(type_save==TYPE_SAVE_REMOTE){
            // 2,異常崩潰信息投遞到服務器
            saveToServer(mContext,ex);
        }
        // 3,應用準備退出
        showToast(mContext, "很抱歉,程序發生異常,即將推出.");
        try {
            Thread.sleep(3500);
        } catch (InterruptedException e) {
            e.printStackTrace();
        }
        ManagerActivity.getInstance().finishActivity();
        android.os.Process.killProcess(android.os.Process.myPid());

    }

    /**
     * 設置自定異常處理類
     *
     * @param pContext
     */
    public void setCustomCrashInfo(Context pContext) {
        this.mContext = pContext;
        Thread.setDefaultUncaughtExceptionHandler(this);
    }
    /**
     * 保存異常信息到sdcard中
     *
     * @param pContext
     * @param ex
     *            異常信息對象
     */
    private void saveToSdcard(Context pContext, Throwable ex) {
        String fileName = null;
        StringBuffer sBuffer = new StringBuffer();
        // 添加異常信息
        sBuffer.append(getExceptionInfo(ex));
        if (Environment.getExternalStorageState().equals(
                Environment.MEDIA_MOUNTED)) {
            File file1 = new File(CRASH_SAVE_SDPATH);
            if (!file1.exists()) {
                file1.mkdir();
            }
            fileName = file1.toString() + File.separator + paserTime(System.currentTimeMillis()) + ".log";
            File file2 = new File(fileName);
            FileOutputStream fos;
            try {
                fos = new FileOutputStream(file2);
                fos.write(sBuffer.toString().getBytes());
                fos.flush();
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
    }

    /**
     * 進行把數據投遞至服務器
     * @param pContext
     * @param ex  崩潰異常
     */
    private void saveToServer(Context pContext,Throwable ex){
        final String carsh_log=getExceptionInfo(ex);
        new Thread(new Runnable() {
            @Override
            public void run() {
                HashMap<String,String> params=new HashMap<String,String>();
                params.put("crash_log",carsh_log);
                String result= IoUtils.responseFromServiceByGetNo(CARSH_LOG_DELIVER, params);
                if (result.equals("1")){
                    Log.d(TAG,"崩潰日志投遞成功...");
                }else {
                    Log.d(TAG,"崩潰日志投遞失敗...");
                }
            }
        }).start();
    }
    /**
     * 獲取并且轉化異常信息
     * 同時可以進行投遞相關的設備,用戶信息
     * @param ex
     * @return 異常信息的字符串形式
     */
    private String getExceptionInfo(Throwable ex) {
        StringWriter sw = new StringWriter();
        ex.printStackTrace(new PrintWriter(sw));
        StringBuffer stringBuffer=new StringBuffer();
        stringBuffer.append("---------Crash Log Begin---------\n");
        //在這邊可以進行相關設備信息投遞--這邊就稍微設置幾個吧
        //其他設備和用戶信息大家可以自己去擴展收集上傳投遞
        stringBuffer.append("SystemVersion:"+ StrUtils.getLocalSystemVersion()+"\n");
        stringBuffer.append(sw.toString()+"\n");
        stringBuffer.append("---------Crash Log End---------\n");
        return stringBuffer.toString();
    }
    /**
     * 進行彈出框提示
     *
     * @param pContext
     * @param msg
     */
    private void showToast(final Context pContext, final String msg) {
        new Thread(new Runnable() {
            @Override
            public void run() {
                Looper.prepare();
                Toast.makeText(pContext, msg, Toast.LENGTH_SHORT).show();
                Looper.loop();
            }
        }).start();
    }
    /**
     * 將毫秒數轉換成yyyy-MM-dd-HH-mm-ss的格式
     * @param milliseconds
     * @return
     */
    private String paserTime(long milliseconds) {
        System.setProperty("user.timezone", "Asia/Shanghai");
        TimeZone tz = TimeZone.getTimeZone("Asia/Shanghai");
        TimeZone.setDefault(tz);
        SimpleDateFormat format = new SimpleDateFormat("yyyy-MM-dd-HH-mm-ss");
        String times = format.format(new Date(milliseconds));

        return times;
    }
}

到此為止我們今天自定義異常捕捉CustomCrash的講解和使用結果,
詳細代碼項目地址: https://github.com/jiangqqlmj/FastDev4Android
同時歡迎大家star和fork整個開源快速開發框架項目~如果有什么意見和反饋,歡迎留言,必定第一時間回復。也歡迎有同樣興趣的童鞋加入到該項目中來,一起維護該項目。

最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。
  • 序言:七十年代末,一起剝皮案震驚了整個濱河市,隨后出現的幾起案子,更是在濱河造成了極大的恐慌,老刑警劉巖,帶你破解...
    沈念sama閱讀 228,030評論 6 531
  • 序言:濱河連續發生了三起死亡事件,死亡現場離奇詭異,居然都是意外死亡,警方通過查閱死者的電腦和手機,發現死者居然都...
    沈念sama閱讀 98,310評論 3 415
  • 文/潘曉璐 我一進店門,熙熙樓的掌柜王于貴愁眉苦臉地迎上來,“玉大人,你說我怎么就攤上這事。” “怎么了?”我有些...
    開封第一講書人閱讀 175,951評論 0 373
  • 文/不壞的土叔 我叫張陵,是天一觀的道長。 經常有香客問我,道長,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 62,796評論 1 309
  • 正文 為了忘掉前任,我火速辦了婚禮,結果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己,他們只是感情好,可當我...
    茶點故事閱讀 71,566評論 6 407
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著,像睡著了一般。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發上,一...
    開封第一講書人閱讀 55,055評論 1 322
  • 那天,我揣著相機與錄音,去河邊找鬼。 笑死,一個胖子當著我的面吹牛,可吹牛的內容都是我干的。 我是一名探鬼主播,決...
    沈念sama閱讀 43,142評論 3 440
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼!你這毒婦竟也來了?” 一聲冷哼從身側響起,我...
    開封第一講書人閱讀 42,303評論 0 288
  • 序言:老撾萬榮一對情侶失蹤,失蹤者是張志新(化名)和其女友劉穎,沒想到半個月后,有當地人在樹林里發現了一具尸體,經...
    沈念sama閱讀 48,799評論 1 333
  • 正文 獨居荒郊野嶺守林人離奇死亡,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內容為張勛視角 年9月15日...
    茶點故事閱讀 40,683評論 3 354
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發現自己被綠了。 大學時的朋友給我發了我未婚夫和他白月光在一起吃飯的照片。...
    茶點故事閱讀 42,899評論 1 369
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖,靈堂內的尸體忽然破棺而出,到底是詐尸還是另有隱情,我是刑警寧澤,帶...
    沈念sama閱讀 38,409評論 5 358
  • 正文 年R本政府宣布,位于F島的核電站,受9級特大地震影響,放射性物質發生泄漏。R本人自食惡果不足惜,卻給世界環境...
    茶點故事閱讀 44,135評論 3 347
  • 文/蒙蒙 一、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧,春花似錦、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 34,520評論 0 26
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽。三九已至,卻和暖如春,著一層夾襖步出監牢的瞬間,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 35,757評論 1 282
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機就差點兒被人妖公主榨干…… 1. 我叫王不留,地道東北人。 一個月前我還...
    沈念sama閱讀 51,528評論 3 390
  • 正文 我出身青樓,卻偏偏與公主長得像,于是被迫代替她去往敵國和親。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當晚...
    茶點故事閱讀 47,844評論 2 372

推薦閱讀更多精彩內容

  • Android 自定義View的各種姿勢1 Activity的顯示之ViewRootImpl詳解 Activity...
    passiontim閱讀 172,613評論 25 708
  • Spring Cloud為開發人員提供了快速構建分布式系統中一些常見模式的工具(例如配置管理,服務發現,斷路器,智...
    卡卡羅2017閱讀 134,781評論 18 139
  • —小荻老師在選擇自己大本營分享整理 1 本次分享的核心要點是: 2 我們遇到的80%以上的問題基本上是溝通問題,我...
    心聲樹洞閱讀 1,117評論 0 1
  • 前段時間看文章的時候,有看到作者推薦了《非暴力溝通》這本書,后來又被其他寫作者推薦過,心想應該是本不錯的書,加之我...
    希亞閱讀 329評論 0 0
  • 當窗前不再映著皎潔的明月。當歲月悄然靜止。。。許。。。浮世奢華也只是一際云煙。若年華倒數。我仰望墨色蒼穹,億萬星辰...
    程子寒閱讀 510評論 6 4