美文网首页
Android APP的异常保存并上传到服务器的处理类

Android APP的异常保存并上传到服务器的处理类

作者: CaoMeng | 来源:发表于2017-11-01 17:24 被阅读114次

确实有同感,我的测试同事测试APP的时候经常说有BUG,但是BUG出现是随机的,不是很稳定,有时会出现,有时不会出现,这个弄得不知何从查找BUG,所以萌发将报错信息获得出来,并保存待下一次启动APP的时候上传至服务器,分析出现的BUG,查找原因修复BUG。本文借鉴于原著Drren

package yan.chen.com.baselibrary.ExpectionCrashHandler;
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.FileInputStream;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
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;
/**
 * Version 1.0
 * Description: 单例的设计模式的异常捕捉
 */
public class ExceptionCrashHandler implements Thread.UncaughtExceptionHandler{
    private static final String TAG = "Tag=ExceptionCrashHandler";
    private static ExceptionCrashHandler mInstance;
    private Context mContext;// 用来获取应用的一些信息
    private Thread.UncaughtExceptionHandler mDefaultExceptionHandler;// 获取系统默认的一场处理
    public static ExceptionCrashHandler getInstance() {
        if (mInstance == null) {
            // 用来解决多并发的问题
            synchronized (ExceptionCrashHandler.class) {
                if (mInstance == null) {
                    mInstance = new ExceptionCrashHandler();
                }
            }
        }
        return mInstance;
    }
    public void init(Context context) {
        this.mContext = context;
        // 设置全局的异常类为本类
        Thread.currentThread().setUncaughtExceptionHandler(this);
        mDefaultExceptionHandler = Thread.currentThread().getDefaultUncaughtExceptionHandler();
    }
    @Override
    public void uncaughtException(Thread thread, Throwable ex) {
        // 全局异常
        Log.e(TAG, "报异常了~!~");
        // 写入到本地文件   ex   当前的版本  手机信息
        // 1. 崩溃的详细信息
        // 2. 应用信息 包名 版本号
        // 3. 手机信息
        // 4.保存当前文件,等应用再次启动再上传,(上传文件不在这里处理)
        String crashFileName = saveInfoToSD(ex);
        Log.e(TAG, "fileName --> " + crashFileName);
        //缓存崩溃日志文件
        cacheCrashFile(crashFileName);
        // 让系统默认处理
        mDefaultExceptionHandler.uncaughtException(thread, 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);
    }

    /**
     * 保存获取的 软件信息,设备信息和出错信息保存在SDcard中
     *
     * @param ex
     * @return
     */
    private String saveInfoToSD(Throwable ex) {
        String fileName = null;
        StringBuffer sb = new StringBuffer();
        // 1. 手机信息 + 应用信息   --> 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.崩溃的详细信息
        sb.append(obtainExceptionInfo(ex));
        // 保存文件  手机应用的目录,并没有拿手机sdCard目录, 6.0 以上需要动态申请权限
        if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {
            File dir = new File(mContext.getFilesDir() + File.separator + "crash" + File.separator);
            // 先删除之前的异常信息
            if (dir.exists()) {
                // 删除该目录下的所有子文件
                deleteDir(dir);
            }
            // 再从新创建文件夹
            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);
    }

    /**
     * 获取一些简单的信息,软件版本,手机版本,型号等信息存放在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;
    }

    /**
     * 获取手机信息  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();
    }


    /**
     * 获取系统未捕捉的错误信息
     * @param throwable
     * @return
     */
    private String obtainExceptionInfo(Throwable throwable) {
        // Java基础 异常
        StringWriter stringWriter = new StringWriter();
        PrintWriter printWriter = new PrintWriter(stringWriter);
        throwable.printStackTrace(printWriter);
        printWriter.close();
        return stringWriter.toString();
    }


    /**
     * 删除目录下的所有文件及子目录下所有文件
     */
    private boolean deleteDir(File dir) {
        if (dir.isDirectory()) {
            File[] children = dir.listFiles();
            // 递归删除目录中的子目录下
            for (File child : children) {
                child.delete();
            }
        }
        // 目录此时为空,可以删除
        return true;
    }
}

这个异常处理类提供获取存储异常信息的文件,可将异常信息读取出来,下面是异常信息的读取方式。读取出来之后,通过服务器接口地址可以将异常信息上传至服务器。

    /*获取崩溃日志的信息的标准写法*/
    File crashFile = ExceptionCrashHandler.getInstance().getCrashFile();
    FileInputStream fis =null;
    try {
        fis =  new FileInputStream(crashFile);
        byte[] buffer = new byte[1024];
        StringBuilder sb = new StringBuilder();
        while (fis.read(buffer) != -1) {
            sb.append(new String(buffer));
        }
        Log.e(TAG,sb.toString()+"在Mainactivity里面测试");
    } catch (FileNotFoundException e) {
        e.printStackTrace();
    } catch (IOException e) {
        e.printStackTrace();
    }finally {
        if (fis!=null){
            fis.close();
        }
    }

最后一件事:需要在APP的Application配置一下该异常类的初始化。

package com.chen.yan;
import android.app.Application;
import android.content.pm.PackageManager;
import com.alipay.euler.andfix.patch.PatchManager;
import yan.chen.com.baselibrary.ExpectionCrashHandler.ExceptionCrashHandler;
/**
 * Created by chen on 2017/10/31.
 */
public class BaseApplication extends Application {
    // Patch管理类
    public static PatchManager mPatchManager;
    @Override
    public void onCreate() {
        super.onCreate();
        // 设置全局异常捕捉类
        ExceptionCrashHandler.getInstance().init(this);
    }
}

配置好之后,当APP报异常的时候,那么就可以保存异常,并读取出来,上传至服务器。

相关文章

网友评论

      本文标题:Android APP的异常保存并上传到服务器的处理类

      本文链接:https://www.haomeiwen.com/subject/fcvwpxtx.html