• android 程序漰溃 后台handle处理类


    [代码] [Java]代码

    001 import java.io.File;
    002 import java.io.FileOutputStream;
    003 import java.io.FilenameFilter;
    004 import java.io.PrintWriter;
    005 import java.io.StringWriter;
    006 import java.io.Writer;
    007 import java.lang.Thread.UncaughtExceptionHandler;
    008 import java.lang.reflect.Field;
    009 import java.util.Arrays;
    010 import java.util.Properties;
    011 import java.util.TreeSet;
    012
    013 import android.content.Context;
    014 import android.content.pm.PackageInfo;
    015 import android.content.pm.PackageManager;
    016 import android.content.pm.PackageManager.NameNotFoundException;
    017 import android.os.Build;
    018 import android.os.Looper;
    019 import android.util.Log;
    020 import android.widget.Toast;
    021
    022 /**
    023 * UncaughtException处理类,当程序发生Uncaught异常的时候,有该类 来接管程序,并记录 发送错误报告.. 註冊方式
    024 * CrashHandler crashHandler = CrashHandler.getInstance(); //注册crashHandler
    025 * crashHandler.init(getApplicationContext()); //发送以前没发送的报告(可选)
    026 * crashHandler.sendPreviousReportsToServer();
    027 *
    028 */
    029 public class CrashHandler implements UncaughtExceptionHandler {
    030 /** Debug Log tag */
    031 public static final String TAG = "CrashHandler";
    032 /**
    033 * 是否开启日志输出,在Debug状态下开启, 在Release状态下关闭以提示程序性能
    034 * */
    035 public static final boolean DEBUG = true;
    036 /** 系统默认的UncaughtException处理类 */
    037 private Thread.UncaughtExceptionHandler mDefaultHandler;
    038 /** CrashHandler实例 */
    039 private static CrashHandler INSTANCE;
    040 /** 程序的Context对象 */
    041 private Context mContext;
    042
    043 /** 使用Properties来保存设备的信息和错误堆栈信息 */
    044 private Properties mDeviceCrashInfo = new Properties();
    045 private static final String VERSION_NAME = "versionName";
    046 private static final String VERSION_CODE = "versionCode";
    047 private static final String STACK_TRACE = "STACK_TRACE";
    048 /** 错误报告文件的扩展名 */
    049 private static final String CRASH_REPORTER_EXTENSION = ".cr";
    050
    051 /** 保证只有一个CrashHandler实例 */
    052 private CrashHandler() {
    053 }
    054
    055 /** 获取CrashHandler实例 ,单例模式 */
    056 public static CrashHandler getInstance() {
    057 if (INSTANCE == null) {
    058 INSTANCE = new CrashHandler();
    059 }
    060 return INSTANCE;
    061 }
    062
    063 /**
    064 * 初始化,注册Context对象, 获取系统默认的UncaughtException处理器, 设置该CrashHandler为程序的默认处理器
    065 *
    066 * @param ctx
    067 */
    068 public void init(Context ctx) {
    069 mContext = ctx;
    070 mDefaultHandler = Thread.getDefaultUncaughtExceptionHandler();
    071 Thread.setDefaultUncaughtExceptionHandler(this);
    072 }
    073
    074 /**
    075 * 当UncaughtException发生时会转入该函数来处理
    076 */
    077 @Override
    078 public void uncaughtException(Thread thread, Throwable ex) {
    079 if (!handleException(ex) && mDefaultHandler != null) {
    080 // 如果用户没有处理则让系统默认的异常处理器来处理
    081 mDefaultHandler.uncaughtException(thread, ex);
    082 } else {
    083 // Sleep一会后结束程序
    084 try {
    085 Thread.sleep(3000);
    086 } catch (InterruptedException e) {
    087 Log.e(TAG, "Error : ", e);
    088 }
    089 android.os.Process.killProcess(android.os.Process.myPid());
    090 System.exit(10);
    091 }
    092 }
    093
    094 /**
    095 * 自定义错误处理,收集错误信息 发送错误报告等操作均在此完成. 开发者可以根据自己的情况来自定义异常处理逻辑
    096 *
    097 * @param ex
    098 * @return true:如果处理了该异常信息;否则返回false
    099 */
    100 private boolean handleException(Throwable ex) {
    101 if (ex == null) {
    102 return true;
    103 }
    104 final String msg = ex.getLocalizedMessage();
    105 // 使用Toast来显示异常信息
    106 new Thread() {
    107 @Override
    108 public void run() {
    109 Looper.prepare();
    110 Toast.makeText(mContext, "程序出错啦:" + msg, Toast.LENGTH_LONG)
    111 .show();
    112 Looper.loop();
    113 }
    114
    115 }.start();
    116 // 收集设备信息
    117 collectCrashDeviceInfo(mContext);
    118 // 保存错误报告文件
    119 String crashFileName = saveCrashInfoToFile(ex);
    120 // 发送错误报告到服务器
    121 sendCrashReportsToServer(mContext);
    122 return true;
    123 }
    124
    125 /**
    126 * 在程序启动时候, 可以调用该函数来发送以前没有发送的报告
    127 */
    128 public void sendPreviousReportsToServer() {
    129 sendCrashReportsToServer(mContext);
    130 }
    131
    132 /**
    133 * 把错误报告发送给服务器,包含新产生的和以前没发送的.
    134 *
    135 * @param ctx
    136 */
    137 private void sendCrashReportsToServer(Context ctx) {
    138 String[] crFiles = getCrashReportFiles(ctx);
    139 if (crFiles != null && crFiles.length > 0) {
    140 TreeSet<String> sortedFiles = new TreeSet<String>();
    141 sortedFiles.addAll(Arrays.asList(crFiles));
    142
    143 for (String fileName : sortedFiles) {
    144 File cr = new File(ctx.getFilesDir(), fileName);
    145 postReport(cr);
    146 cr.delete();// 删除已发送的报告
    147 }
    148 }
    149 }
    150
    151 private void postReport(File file) {
    152 // TODO 使用HTTP Post 发送错误报告到服务器
    153 // 这里不再详述,开发者可以根据OPhoneSDN上的其他网络操作
    154 // 教程来提交错误报告
    155 }
    156
    157 /**
    158 * 获取错误报告文件名
    159 *
    160 * @param ctx
    161 * @return
    162 */
    163 private String[] getCrashReportFiles(Context ctx) {
    164 File filesDir = ctx.getFilesDir();
    165 FilenameFilter filter = new FilenameFilter() {
    166 public boolean accept(File dir, String name) {
    167 return name.endsWith(CRASH_REPORTER_EXTENSION);
    168 }
    169 };
    170 return filesDir.list(filter);
    171 }
    172
    173 /**
    174 * 保存错误信息到文件中
    175 *
    176 * @param ex
    177 * @return
    178 */
    179 private String saveCrashInfoToFile(Throwable ex) {
    180 Writer info = new StringWriter();
    181 PrintWriter printWriter = new PrintWriter(info);
    182 ex.printStackTrace(printWriter);
    183
    184 Throwable cause = ex.getCause();
    185 while (cause != null) {
    186 cause.printStackTrace(printWriter);
    187 cause = cause.getCause();
    188 }
    189
    190 String result = info.toString();
    191 printWriter.close();
    192 mDeviceCrashInfo.put(STACK_TRACE, result);
    193 String fileName = "";
    194 try {
    195 long timestamp = System.currentTimeMillis();
    196 fileName = "crash-" + timestamp + CRASH_REPORTER_EXTENSION;
    197 FileOutputStream trace = mContext.openFileOutput(fileName,
    198 Context.MODE_PRIVATE);
    199 mDeviceCrashInfo.store(trace, "");
    200 trace.flush();
    201 trace.close();
    202 return fileName;
    203 } catch (Exception e) {
    204 Log.e(TAG, "an error occured while writing report file..."
    205 + fileName, e);
    206 }
    207 return null;
    208 }
    209
    210 /**
    211 * 收集程序崩溃的设备信息
    212 *
    213 * @param ctx
    214 */
    215 public void collectCrashDeviceInfo(Context ctx) {
    216 try {
    217 PackageManager pm = ctx.getPackageManager();
    218 PackageInfo pi = pm.getPackageInfo(ctx.getPackageName(),
    219 PackageManager.GET_ACTIVITIES);
    220 if (pi != null) {
    221 mDeviceCrashInfo.put(VERSION_NAME,
    222 pi.versionName == null ? "not set" : pi.versionName);
    223 mDeviceCrashInfo.put(VERSION_CODE, pi.versionCode);
    224 }
    225 } catch (NameNotFoundException e) {
    226 Log.e(TAG, "Error while collect package info", e);
    227 }
    228 // 使用反射来收集设备信息.在Build类中包含各种设备信息,
    229 // 例如: 系统版本号,设备生产商 等帮助调试程序的有用信息
    230 // 具体信息请参考后面的截图
    231 Field[] fields = Build.class.getDeclaredFields();
    232 for (Field field : fields) {
    233 try {
    234 field.setAccessible(true);
    235 mDeviceCrashInfo.put(field.getName(), field.get(null));
    236 if (DEBUG) {
    237 Log.d(TAG, field.getName() + " : " + field.get(null));
    238 }
    239 } catch (Exception e) {
    240 Log.e(TAG, "Error while collect crash info", e);
    241 }
    242
    243 }
    244
    245 }
    246
    247 }
  • 相关阅读:
    阿里开源混沌工程工具 ChaosBlade
    十天入门java教程 Day01
    如何破解IDEA
    Locust压力测试使用总结
    python+requests接口自动化测试框架实例详解教程
    一个完整的性能测试流程
    jmeter测试报告汉化及脚本编写
    tomcat的日志文件权限与启动用户的权限不一致
    Linux下部署开源版“禅道”项目管理系统
    ELK原理与介绍
  • 原文地址:https://www.cnblogs.com/tuncaysanli/p/2469042.html
Copyright © 2020-2023  润新知