android全局捕獲異常使用詳解

  1. 使用原因
    在開發(fā)過(guò)程中一定會(huì)遇到程序異常、崩潰、crash猬错,有時(shí)候用戶隨便點(diǎn)擊導(dǎo)致崩潰,此時(shí)我們無(wú)法及時(shí)收集崩潰信息茸歧,可以用全局捕獲異常來(lái)收集崩潰信息,將其存入本地目錄或者上傳至服務(wù)器显沈,每次進(jìn)入頁(yè)面[MainActivity]時(shí)软瞎,獲取該文件,然后打印出其結(jié)果拉讯,就可以定位到具體哪一行報(bào)錯(cuò)涤浇,直接解決即可
  2. 使用步驟
    2.1:自定義ExceptionCrashHandler
/**
 * @email :  2185134304@qq.com
 * @date :2017/12/23
 * @author : Jack-Chen
 * @Description: 單例的設(shè)計(jì)模式的異常捕捉
 */
public class ExceptionCrashHandler implements Thread.UncaughtExceptionHandler{

    private static final String TAG = "ExceptionCrashHandler" ;


    //獲取系統(tǒng)默認(rèn)的 線程未捕獲異常
    private Thread.UncaughtExceptionHandler mDefaultExceptionHandler ;

    private static ExceptionCrashHandler mInstance ;

    public static ExceptionCrashHandler getmInstance(){

        if (mInstance == null){
            //用來(lái)解決多并發(fā)問(wèn)題
            synchronized (ExceptionCrashHandler.class){
                if (mInstance == null){
                    mInstance = new ExceptionCrashHandler() ;
                }
            }

        }

        return mInstance;
    }


    //獲取應(yīng)用的一些信息
    private Context mContext ;

    public void init(Context context){
        this.mContext = context ;

        //設(shè)置全局的異常類為本類
        Thread.currentThread().setUncaughtExceptionHandler(this);

        mDefaultExceptionHandler = Thread.currentThread().getUncaughtExceptionHandler() ;
    }


    /**
     * 只要發(fā)生異常,就會(huì)調(diào)用此方法魔慷,此方法為全局異常
     * @param thread
     * @param throwable
     */
    @Override
    public void uncaughtException(Thread thread, Throwable throwable) {
        Log.e(TAG , "報(bào)異常了.......") ;


        // 寫入到本地文件   ex   當(dāng)前的版本  手機(jī)信息


        // 1. 崩潰的詳細(xì)信息


        // 2. 應(yīng)用信息 包名 版本號(hào)


        // 3. 手機(jī)信息


        // 4.保存當(dāng)前文件只锭,等應(yīng)用再次啟動(dòng)再上傳,(上傳文件不在這里處理)

        String crashFileName = saveInfoToSD(throwable);

        Log.e(TAG, "fileName --> " + crashFileName);

        // 3. 緩存崩潰日志文件
        cacheCrashFile(crashFileName);



        //把異常寫入到本地文件  ex[比如] 當(dāng)前的版本信息院尔、手機(jī)信息
        mDefaultExceptionHandler.uncaughtException(thread , throwable);

    }


    /**
     * 緩存崩潰日志文件 用SP存儲(chǔ)崩潰文件到本地
     *
     * @param fileName
     */
    private void cacheCrashFile(String fileName) {
        SharedPreferences sp = mContext.getSharedPreferences("crash", Context.MODE_PRIVATE);
//        sp.edit().putString("CRASH_FILE_NAME", fileName).commit();
        sp.edit().putString("TJTREK_CRASH_FILE_NAME", fileName).commit();
    }

    /**
     * 獲取崩潰文件名稱
     *
     * @return
     */
    public File getCrashFile() {
        String crashFileName = mContext.getSharedPreferences("crash",
                Context.MODE_PRIVATE).getString("TJTREK_CRASH_FILE_NAME", ""); //CRASH_FILE_NAME
        return new File(crashFileName);
    }

    /**
     * 保存獲取的 軟件信息蜻展,設(shè)備信息和出錯(cuò)信息保存在SDcard中
     *
     * @param ex
     * @return
     */
    private String saveInfoToSD(Throwable ex) {
        String fileName = null;
        StringBuffer sb = new StringBuffer();

        // 1. 手機(jī)信息 + 應(yīng)用信息   --> obtainSimpleInfo()
        for (Map.Entry<String, String> entry : obtainSimpleInfo(mContext)
                .entrySet()) {
            String key = entry.getKey();
            String value = entry.getValue();
            sb.append(key).append(" = ").append(value).append("\n");
        }

        // 2.崩潰的詳細(xì)信息
        sb.append(obtainExceptionInfo(ex));

        // 保存文件  手機(jī)應(yīng)用的目錄,并沒有拿手機(jī)sdCard目錄邀摆, 6.0 以上需要?jiǎng)討B(tài)申請(qǐng)權(quán)限

        if (Environment.getExternalStorageState().equals(
                Environment.MEDIA_MOUNTED)) {

            File dir = new File(mContext.getFilesDir() + File.separator + "crash"
                    + File.separator);

            // 先刪除之前的異常信息
            if (dir.exists()) {
                // 刪除該目錄下的所有子文件
                deleteDir(dir);
            }

            // 再?gòu)男聞?chuàng)建文件夾
            if (!dir.exists()) {
                dir.mkdir();
            }

            try {
                fileName = dir.toString()
                        + File.separator
                        + getAssignTime("yyyy_MM_dd_HH_mm") + ".txt";
                FileOutputStream fos = new FileOutputStream(fileName);
                fos.write(sb.toString().getBytes());
                fos.flush();
                fos.close();
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
        return fileName;
    }

    private String getAssignTime(String dateFormatStr) {
        DateFormat dataFormat = new SimpleDateFormat(dateFormatStr);
        long currentTime = System.currentTimeMillis();
        return dataFormat.format(currentTime);
    }

    /**
     * 獲取一些簡(jiǎn)單的信息,軟件版本纵顾,手機(jī)版本,型號(hào)等信息存放在HashMap中
     *
     * @return
     */
    private HashMap<String, String> obtainSimpleInfo(Context context) {
        HashMap<String, String> map = new HashMap<>();
        PackageManager mPackageManager = context.getPackageManager();
        PackageInfo mPackageInfo = null;
        try {
            mPackageInfo = mPackageManager.getPackageInfo(
                    context.getPackageName(), PackageManager.GET_ACTIVITIES);
        } catch (PackageManager.NameNotFoundException e) {
            e.printStackTrace();
        }
        map.put("versionName", mPackageInfo.versionName);
        map.put("versionCode", "" + mPackageInfo.versionCode);
        map.put("MODEL", "" + Build.MODEL);
        map.put("SDK_INT", "" + Build.VERSION.SDK_INT);
        map.put("PRODUCT", "" + Build.PRODUCT);
        map.put("MOBLE_INFO", getMobileInfo());
        return map;
    }

    /**
     * 獲取手機(jī)信息  HomiNote 6.0
     *
     * @return
     */
    public static String getMobileInfo() {
        StringBuffer sb = new StringBuffer();
        try {
            // 利用反射獲取 Build 的所有屬性
            Field[] fields = Build.class.getDeclaredFields();
            for (Field field : fields) {
                field.setAccessible(true);
                String name = field.getName();
                String value = field.get(null).toString();
                sb.append(name + "=" + value);
                sb.append("\n");
            }
        } catch (Exception e) {
            e.printStackTrace();
        }
        return sb.toString();
    }


    /**
     * 獲取系統(tǒng)未捕捉的錯(cuò)誤信息
     *
     * @param throwable
     * @return
     */
    private String obtainExceptionInfo(Throwable throwable) {
        // Java基礎(chǔ) 異常
        StringWriter stringWriter = new StringWriter();
        PrintWriter printWriter = new PrintWriter(stringWriter);
        throwable.printStackTrace(printWriter);
        printWriter.close();
        return stringWriter.toString();
    }


    /**
     * 遞歸刪除目錄下的所有文件及子目錄下所有文件
     *
     * @param dir 將要?jiǎng)h除的文件目錄
     * @return boolean Returns "true" if all deletions were successful. If a
     * deletion fails, the method stops attempting to delete and returns
     * "false".
     */
    private boolean deleteDir(File dir) {
        if (dir.isDirectory()) {
            File[] children = dir.listFiles();
            // 遞歸刪除目錄中的子目錄下
            for (File child : children) {
                child.delete();
            }
        }
        // 目錄此時(shí)為空栋盹,可以刪除
        return true;
    }
}

2.2: 在自己程序中BaseApplication中的onCreate()方法設(shè)置全局異常捕捉類

/**
 * @date : 2014/10/18
 * @author Jack-Chen
 * @function:
 *      自定義的Application 1施逾、運(yùn)行的過(guò)程中只有一個(gè)實(shí)例 2、一個(gè)應(yīng)用程序最先執(zhí)行的方法 運(yùn)行在主線程中
 *      注意:在AndroidManifest文件中進(jìn)行注冊(cè)
 */
public class BaseApplication extends Application {
    @Override
    public void onCreate() {
        super.onCreate();
        //設(shè)置全局異常捕捉類
        ExceptionCrashHandler.getmInstance().init(this);
    }
}

2.3:直接在MainActivity的initData()初始化數(shù)據(jù)方法中例获,獲取上次崩潰信息汉额,然后打印即可

    @Override
    public void initData() {
        //每次運(yùn)行程序都會(huì)獲取上次的崩潰信息
        File crashFile = ExceptionCrashHandler.getmInstance().getCrashFile() ;
        if (crashFile.exists()){
            //將保存的崩潰日志讀取出來(lái)并打印,就知道崩潰具體信息榨汤,然后處理
            try {
                InputStreamReader fileReader = new InputStreamReader(new FileInputStream(crashFile)) ;
                char[] buffer = new char[1024] ;
                int len=0;
                while ((len=fileReader.read(buffer))!=-1){
                    String str = new String(buffer , 0 , len);
                    Log.d("=====崩潰信息打印====" , str+"--------") ;
                }
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
}

以上就是全局異常捕獲及使用步驟蠕搜,如若需要,直接拷貝到自己項(xiàng)目中即可使用

如果對(duì)您有幫助收壕,可以關(guān)注我的簡(jiǎn)書讥脐,我會(huì)持續(xù)更新文章遭居,將自己項(xiàng)目中出現(xiàn)的問(wèn)題、bug旬渠、及解決方法都會(huì)上傳簡(jiǎn)書俱萍,我不希望別人去踩我自己踩過(guò)的坑,因?yàn)槲矣X得這種東西沒必要所有人都去犯告丢,希望自己綿薄之力對(duì)您有幫助

地址如下
http://www.reibang.com/u/c5fabe27176e

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請(qǐng)聯(lián)系作者
  • 序言:七十年代末枪蘑,一起剝皮案震驚了整個(gè)濱河市,隨后出現(xiàn)的幾起案子岖免,更是在濱河造成了極大的恐慌岳颇,老刑警劉巖,帶你破解...
    沈念sama閱讀 212,884評(píng)論 6 492
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件颅湘,死亡現(xiàn)場(chǎng)離奇詭異话侧,居然都是意外死亡,警方通過(guò)查閱死者的電腦和手機(jī)闯参,發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 90,755評(píng)論 3 385
  • 文/潘曉璐 我一進(jìn)店門瞻鹏,熙熙樓的掌柜王于貴愁眉苦臉地迎上來(lái),“玉大人鹿寨,你說(shuō)我怎么就攤上這事新博。” “怎么了脚草?”我有些...
    開封第一講書人閱讀 158,369評(píng)論 0 348
  • 文/不壞的土叔 我叫張陵赫悄,是天一觀的道長(zhǎng)。 經(jīng)常有香客問(wèn)我馏慨,道長(zhǎng)埂淮,這世上最難降的妖魔是什么? 我笑而不...
    開封第一講書人閱讀 56,799評(píng)論 1 285
  • 正文 為了忘掉前任写隶,我火速辦了婚禮同诫,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘樟澜。我一直安慰自己误窖,他們只是感情好,可當(dāng)我...
    茶點(diǎn)故事閱讀 65,910評(píng)論 6 386
  • 文/花漫 我一把揭開白布秩贰。 她就那樣靜靜地躺著霹俺,像睡著了一般。 火紅的嫁衣襯著肌膚如雪毒费。 梳的紋絲不亂的頭發(fā)上丙唧,一...
    開封第一講書人閱讀 50,096評(píng)論 1 291
  • 那天,我揣著相機(jī)與錄音觅玻,去河邊找鬼想际。 笑死培漏,一個(gè)胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的胡本。 我是一名探鬼主播牌柄,決...
    沈念sama閱讀 39,159評(píng)論 3 411
  • 文/蒼蘭香墨 我猛地睜開眼,長(zhǎng)吁一口氣:“原來(lái)是場(chǎng)噩夢(mèng)啊……” “哼侧甫!你這毒婦竟也來(lái)了珊佣?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 37,917評(píng)論 0 268
  • 序言:老撾萬(wàn)榮一對(duì)情侶失蹤披粟,失蹤者是張志新(化名)和其女友劉穎咒锻,沒想到半個(gè)月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體守屉,經(jīng)...
    沈念sama閱讀 44,360評(píng)論 1 303
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡惑艇,尸身上長(zhǎng)有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 36,673評(píng)論 2 327
  • 正文 我和宋清朗相戀三年,在試婚紗的時(shí)候發(fā)現(xiàn)自己被綠了拇泛。 大學(xué)時(shí)的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片滨巴。...
    茶點(diǎn)故事閱讀 38,814評(píng)論 1 341
  • 序言:一個(gè)原本活蹦亂跳的男人離奇死亡,死狀恐怖碰镜,靈堂內(nèi)的尸體忽然破棺而出,到底是詐尸還是另有隱情习瑰,我是刑警寧澤绪颖,帶...
    沈念sama閱讀 34,509評(píng)論 4 334
  • 正文 年R本政府宣布甜奄,位于F島的核電站柠横,受9級(jí)特大地震影響,放射性物質(zhì)發(fā)生泄漏课兄。R本人自食惡果不足惜牍氛,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 40,156評(píng)論 3 317
  • 文/蒙蒙 一搬俊、第九天 我趴在偏房一處隱蔽的房頂上張望。 院中可真熱鬧蜒茄,春花似錦唉擂、人聲如沸。這莊子的主人今日做“春日...
    開封第一講書人閱讀 30,882評(píng)論 0 21
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽(yáng)。三九已至屿聋,卻和暖如春空扎,著一層夾襖步出監(jiān)牢的瞬間藏鹊,已是汗流浹背。 一陣腳步聲響...
    開封第一講書人閱讀 32,123評(píng)論 1 267
  • 我被黑心中介騙來(lái)泰國(guó)打工转锈, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留盘寡,地道東北人。 一個(gè)月前我還...
    沈念sama閱讀 46,641評(píng)論 2 362
  • 正文 我出身青樓黑忱,卻偏偏與公主長(zhǎng)得像宴抚,于是被迫代替她去往敵國(guó)和親。 傳聞我的和親對(duì)象是個(gè)殘疾皇子甫煞,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 43,728評(píng)論 2 351