您的位置:首页 > 移动开发 > Android开发

使用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.下面是我自己实现的一个异常处理器:

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的博客啦,希望对你有所帮助,谢谢观看!感谢其他博主的分享!
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息