使用CrashHandler来获取(收集)应用的crash信息
2017-12-22 11:11
471 查看
1.Android应用不可避免地发生crash,也称之为崩溃,无论你的程序写得多完美,总是无法完全避免crash的发生,可能是由于Android系统底层的bug,也可能是由于不充分的机型适配或者网络状况。当crash发生时,系统会kill掉正在执行的程序,现象就是闪退或者提示用户已停止运行。这对用户来说是很不友好的,也是开发者所不愿看到的,更糟糕的是,当用户发生了crash,(由于产品发布的原因)开发者却无法得知程序为何crash。
2.Android提供了处理这类问题的方法。Thread类的一个方法setDefaultUncaughtExceptionHandler()。这个方法可以设置系统的默认异常处理器。当crash发生时,系统就会回调UncaughtExceptionHandler的uncaughtException方法,在uncaughtException方法中就可以获取到异常信息(代码中捕获到的异常时不会讲给CrashHandler处理的),可以选择把异常信息存储到SD卡上,然后在合适的时机通过网络将crash信息上传到服务器上,这样开发人员就可以分析用户crash的场景从而在后面的版本中修复此类crash。
3.从上面的分析可以知道,获取应用的crash的信息的方式,首先要实现一个UncaughtExceptionHandler对象,在它的uncaughtException方法中获取异常信息并将其存储在SD卡中(在适当的时候上传到服务器上供开发人员分析),然后调用Thread的setDefaultUncaughtExceptionHandler()将它设置为线程默认异常处理器(可以选择在Application初始化的时候为程序设置CrashHandler),由于默认义仓处理器时Thread类的静态成员,因此它的作用对象时当前进程的所有线程。
4.下面是我自己实现的一个异常处理器:
5.那该什么时候为线程设置CrashHandler呢,很简单,在Application初始化的时候便可。
以上就是所有关于CrashHandler的博客啦,希望对你有所帮助,谢谢观看!感谢其他博主的分享!
2.Android提供了处理这类问题的方法。Thread类的一个方法setDefaultUncaughtExceptionHandler()。这个方法可以设置系统的默认异常处理器。当crash发生时,系统就会回调UncaughtExceptionHandler的uncaughtException方法,在uncaughtException方法中就可以获取到异常信息(代码中捕获到的异常时不会讲给CrashHandler处理的),可以选择把异常信息存储到SD卡上,然后在合适的时机通过网络将crash信息上传到服务器上,这样开发人员就可以分析用户crash的场景从而在后面的版本中修复此类crash。
3.从上面的分析可以知道,获取应用的crash的信息的方式,首先要实现一个UncaughtExceptionHandler对象,在它的uncaughtException方法中获取异常信息并将其存储在SD卡中(在适当的时候上传到服务器上供开发人员分析),然后调用Thread的setDefaultUncaughtExceptionHandler()将它设置为线程默认异常处理器(可以选择在Application初始化的时候为程序设置CrashHandler),由于默认义仓处理器时Thread类的静态成员,因此它的作用对象时当前进程的所有线程。
4.下面是我自己实现的一个异常处理器:
import android.content.Context; 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.BufferedWriter; import java.io.File; import java.io.FileWriter; import java.io.PrintWriter; import java.io.StringWriter; import java.io.Writer; import java.lang.reflect.Field; import java.text.SimpleDateFormat; import java.util.Date; import java.util.HashMap; import java.util.Map; /** * Created by zb.yang on 2017/12/22. */ public class ZbCrashHandler implements Thread.UncaughtExceptionHandler { private static final String TAG = "ZbCrashHandler"; private static final String PATH = Environment.getExternalStorageDirectory().getPath()+"/CrashHandler/log"; private static final String FILE_NAME = "crash"; private static final String FILE_NAME_SUFFIX = ".trace"; //系统默认的UncaughtException处理类 private Thread.UncaughtExceptionHandler mDefaultHandler; //CrashHandler实例 private static ZbCrashHandler INSTANCE; //程序的Context对象 private Context mContext; //用来存储设备信息和异常信息 private Map<String, String> infos = new HashMap<String, String>(); private ZbCrashHandler(){ } /** * 通过静态类返回单例 */ private static class SingletonHolder { private static final ZbCrashHandler INSTANCE = new ZbCrashHandler(); } public static ZbCrashHandler getINSTANCE(){ return SingletonHolder.INSTANCE; } /** * 初始化 */ public void init(Context context){ mContext = context; //获取系统默认的UncaughtException处理器 mDefaultHandler = Thread.getDefaultUncaughtExceptionHandler(); //设置该CrashHandler为程序的默认处理器 Thread.setDefaultUncaughtExceptionHandler(this); } /** * 当UncaughtException发生时会转入该函数来处理 */ @Override public void uncaughtException(Thread t, Throwable e) { if (!handleException(e) && mDefaultHandler != null) { //如果用户没有处理则让系统默认的异常处理器来处理 mDefaultHandler.uncaughtException(t, e); } else { try { Thread.sleep(3000); } catch (InterruptedException ie) { LogUtils.e(ie.toString()); } //退出程序 android.os.Process.killProcess(android.os.Process.myPid()); System.exit(1); } } /** * 自定义错误处理,收集错误信息 发送错误报告等操作均在此完成. * * @param ex * @return true:如果处理了该异常信息;否则返回false. */ private boolean handleException(Throwable ex) { if (ex == null) { return false; } //收集设备参数信息 collectDeviceInfo(mContext); //保存日志文件 saveCrashInfo2File(ex); return true; } /** * 收集设备参数信息 * * @param ctx */ private void collectDeviceInfo(Context ctx) { try { PackageManager pm = ctx.getPackageManager(); PackageInfo pi = pm.getPackageInfo(ctx.getPackageName(), PackageManager.GET_ACTIVITIES); if (pi != null) { String versionName = pi.versionName == null ? "null" : pi.versionName; String versionCode = pi.versionCode + ""; infos.put("versionName", versionName); infos.put("versionCode", versionCode); } } catch (PackageManager.NameNotFoundException e) { LogUtils.e(TAG,"CrashHandleran.NameNotFoundException---> error occured when collect package info"); } //收集设备型号信息 Field[] fields = Build.class.getDeclaredFields(); for (Field field : fields) { try { field.setAccessible(true); infos.put(field.getName(), field.get(null).toString()); } catch (Exception e) { LogUtils.e(TAG,"CrashHandler.NameNotFoundException---> an error occured when collect crash info"); } } } /** * 保存错误信息到文件中 * * @param ex * @return 返回文件名称, 便于将文件传送到服务器 */ private String saveCrashInfo2File(Throwable ex) { if(!Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) { Log.e(TAG,"sdcard unmounted, skip saveCrashInfo2File"); } //调用这个方法之前在6.0之后需要动态权限请求 File dir = new File(PATH); if(!dir.exists()){ dir.mkdirs(); } long current = System.currentTimeMillis(); String time = new SimpleDateFormat("yyyy-MM-dd HH:mm:ss").format(new Date(current)); String file_name = PATH+FILE_NAME+time+FILE_NAME_SUFFIX; File file = new File(file_name); try{ PrintWriter pw = new PrintWriter(new BufferedWriter(new FileWriter(file))); pw.println(time); StringBuffer sb = new StringBuffer(); for (Map.Entry<String, String> entry : infos.entrySet()) { String key = entry.getKey(); String value = entry.getValue(); sb.append(key + "=" + value + "\n"); } pw.println(sb.toString()); pw.println(); ex.printStackTrace(pw); Throwable cause = ex.getCause(); while (cause != null) { cause.printStackTrace(pw); cause = cause.getCause(); } }catch (Exception e){ Log.e(TAG,"save crash info into file failed"); } return file_name; } public void uploadExceptionToServer(){ // TODO upload Ecxeption Message to Your Web server; } }
5.那该什么时候为线程设置CrashHandler呢,很简单,在Application初始化的时候便可。
public class App extends BaseApplication { @Override public void onCreate() { super.onCreate(); ZbCrashHandler zbCrashHandler = ZbCrashHandler.getINSTANCE(); zbCrashHandler.init(getApplicationContext()); } }
以上就是所有关于CrashHandler的博客啦,希望对你有所帮助,谢谢观看!感谢其他博主的分享!
相关文章推荐
- Android 使用CrashHandler获取应用的crash信息
- 使用CrashHandler来获取应用的crash信息
- 使用CrashHandler来获取应用的Crash信息
- 从框架到完整项目搭建,实战项目《约个球》(2)-框架搭建之使用CrashHandler来获取应用的Crash信息
- Android 使用 CrashHandler 来获取应用的 crash 信息
- 使用CrashHandler来获取应用的crash信息
- Android使用CrashHandler来获取应用的crash信息的方法
- Android 使用CrashHandler获取应用的crash信息
- android 程序崩溃信息的收集【使用CrashHandler来收集应用的crash信息】
- 使用CrashHandler来获取应用的oom crash信息
- 通过CrashHandler获取应用的crash信息
- Android应用Crash信息收集
- Android 全局异常捕获之CrashHandler,用于开发者收集获取异常信息
- javascript 的高级应用 获取浏览器的信息:使用navigator对象
- Android群英传——第九章使用PackageManager获取应用包信息
- Android使用UncaughtExceptionHandler获取crash信息
- Android开发——收集应用Crash信息到服务器
- 实现UncaughtExceptionHandler来实现获取应用全局的crash信息
- 使用 python 收集获取 Linux 系统主机信息
- Android系统获取应用的Crash信息,保存在本地(可以自己实现上传到自己的服务器)