Android——上線APP奔潰之后錯誤日志的抓取

緣由

提供給老外的一個APP測試過程中出現(xiàn)異常,APP奔潰,千里之隔,時差有別,很絕望。。。只能再搞一個抓取日志的功能分析問題所在,基于懶人思想,先查,然后整合總結(jié)。有了下面的東西。

支持原作

我是從這里搬過來的,之后只做一個保留文件處的小修改而已:https://blog.csdn.net/omnispace/article/details/79833862
感謝大佬!感謝大佬!感謝大佬!

小提示

1.類中有一個方法getCrashFile()是用來獲取這個錯誤日志,獲取后你就可以為所欲為。。。
2.CrashHandler.getInstance().init(this);在Application的中調(diào)用。。

直接上代碼

import android.content.Context;
import android.content.SharedPreferences;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.os.Build;
import android.os.Environment;
import android.util.Log;

import java.io.File;
import java.io.FileOutputStream;
import java.io.PrintWriter;
import java.io.StringWriter;
import java.lang.reflect.Field;
import java.text.DateFormat;
import java.text.SimpleDateFormat;
import java.util.HashMap;
import java.util.Map;

/**
* Author: 蔡小樹
* Time: 2018/12/11 17:26
* Description: 自己寫的錯誤日志抓取工具
*/
public class CrashHandler implements Thread.UncaughtExceptionHandler {

    private static CrashHandler crashUtils;
    private Context mContext;
    private Thread.UncaughtExceptionHandler mDefaultHandler;
    private static final String TAG = "CrashHandler";

    private CrashHandler() {
    }

    public static CrashHandler getInstance() {
        if (crashUtils == null) {
            crashUtils = new CrashHandler();
        }
        return crashUtils;
    }

    /**
     * 初始化
     *
     * @param context
     */
    public void init(Context context) {
        mContext = context;
        //獲取系統(tǒng)默認(rèn)的UncaughtException
        mDefaultHandler = Thread.getDefaultUncaughtExceptionHandler();
        //將自己的Crash放進(jìn)去
        Thread.setDefaultUncaughtExceptionHandler(this);
    }

    @Override
    public void uncaughtException(Thread t, Throwable ex) {
        Log.e(TAG, "捕捉到了異常");
        // 1. 獲取信息
        // 1.1 崩潰信息
        // 1.2 手機(jī)信息
        // 1.3 版本信息
        // 2.寫入文件
        String crashFileName = saveInfoToSD(ex);
        Log.e(TAG, "fileName --> " + crashFileName);

        // 3. 緩存崩潰日志文件
        cacheCrashFile(crashFileName);
        // 系統(tǒng)默認(rèn)處理
        mDefaultHandler.uncaughtException(t, ex);
    }

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


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

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

        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");
        }

        sb.append(obtainExceptionInfo(ex));

        if (Environment.getExternalStorageState().equals(
                Environment.MEDIA_MOUNTED)) {
            try {
                //這里我并不需要存很多錯誤日志,只能存一個
                String path = mContext.getExternalFilesDir("log").getPath();
                File dir = new File(path);
                if (!dir.exists()) {
                    dir.mkdirs();
                } else {
                    File[] files = dir.listFiles();
                    for (File file :
                            files) {
                        file.delete();
                    }
                }
                FileOutputStream fos = new FileOutputStream(path + "/error" + getAssignTime("yyyyMMdd-HH:mm") + ".log", true);
                fos.write(sb.toString().getBytes());
                fos.flush();
                fos.close();
            } catch (Exception e) {
                e.printStackTrace();
            }
        }
        return fileName;
    }

    /**
     * 返回當(dāng)前日期根據(jù)格式
     **/
    private String getAssignTime(String dateFormatStr) {
        DateFormat dataFormat = new SimpleDateFormat(dateFormatStr);
        long currentTime = System.currentTimeMillis();
        return dataFormat.format(currentTime);
    }


    /**
     * 獲取一些簡單的信息,軟件版本,手機(jī)版本,型號等信息存放在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;
    }


    /**
     * Cell phone information
     *
     * @return
     */
    public static String getMobileInfo() {
        StringBuffer sb = new StringBuffer();
        try {
            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)未捕捉的錯誤信息
     *
     * @param throwable
     * @return
     */
    private String obtainExceptionInfo(Throwable throwable) {
        StringWriter stringWriter = new StringWriter();
        PrintWriter printWriter = new PrintWriter(stringWriter);
        throwable.printStackTrace(printWriter);
        printWriter.close();
        return stringWriter.toString();
    }
}
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
【社區(qū)內(nèi)容提示】社區(qū)部分內(nèi)容疑似由AI輔助生成,瀏覽時請結(jié)合常識與多方信息審慎甄別。
平臺聲明:文章內(nèi)容(如有圖片或視頻亦包括在內(nèi))由作者上傳并發(fā)布,文章內(nèi)容僅代表作者本人觀點(diǎn),簡書系信息發(fā)布平臺,僅提供信息存儲服務(wù)。

相關(guān)閱讀更多精彩內(nèi)容

友情鏈接更多精彩內(nèi)容