- 使用原因
在開發(fā)過(guò)程中一定會(huì)遇到程序異常、崩潰、crash猬错,有時(shí)候用戶隨便點(diǎn)擊導(dǎo)致崩潰,此時(shí)我們無(wú)法及時(shí)收集崩潰信息茸歧,可以用全局捕獲異常來(lái)收集崩潰信息,將其存入本地目錄或者上傳至服務(wù)器显沈,每次進(jìn)入頁(yè)面[MainActivity]時(shí)软瞎,獲取該文件,然后打印出其結(jié)果拉讯,就可以定位到具體哪一行報(bào)錯(cuò)涤浇,直接解決即可 - 使用步驟
2.1:自定義ExceptionCrashHandler
/**
* @email : 2185134304@qq.com
* @date :2017/12/23
* @author : Jack-Chen
* @Description: 單例的設(shè)計(jì)模式的異常捕捉
*/
public class ExceptionCrashHandler implements Thread.UncaughtExceptionHandler{
private static final String TAG = "ExceptionCrashHandler" ;
//獲取系統(tǒng)默認(rèn)的 線程未捕獲異常
private Thread.UncaughtExceptionHandler mDefaultExceptionHandler ;
private static ExceptionCrashHandler mInstance ;
public static ExceptionCrashHandler getmInstance(){
if (mInstance == null){
//用來(lái)解決多并發(fā)問(wèn)題
synchronized (ExceptionCrashHandler.class){
if (mInstance == null){
mInstance = new ExceptionCrashHandler() ;
}
}
}
return mInstance;
}
//獲取應(yīng)用的一些信息
private Context mContext ;
public void init(Context context){
this.mContext = context ;
//設(shè)置全局的異常類為本類
Thread.currentThread().setUncaughtExceptionHandler(this);
mDefaultExceptionHandler = Thread.currentThread().getUncaughtExceptionHandler() ;
}
/**
* 只要發(fā)生異常,就會(huì)調(diào)用此方法魔慷,此方法為全局異常
* @param thread
* @param throwable
*/
@Override
public void uncaughtException(Thread thread, Throwable throwable) {
Log.e(TAG , "報(bào)異常了.......") ;
// 寫入到本地文件 ex 當(dāng)前的版本 手機(jī)信息
// 1. 崩潰的詳細(xì)信息
// 2. 應(yīng)用信息 包名 版本號(hào)
// 3. 手機(jī)信息
// 4.保存當(dāng)前文件只锭,等應(yīng)用再次啟動(dòng)再上傳,(上傳文件不在這里處理)
String crashFileName = saveInfoToSD(throwable);
Log.e(TAG, "fileName --> " + crashFileName);
// 3. 緩存崩潰日志文件
cacheCrashFile(crashFileName);
//把異常寫入到本地文件 ex[比如] 當(dāng)前的版本信息院尔、手機(jī)信息
mDefaultExceptionHandler.uncaughtException(thread , throwable);
}
/**
* 緩存崩潰日志文件 用SP存儲(chǔ)崩潰文件到本地
*
* @param fileName
*/
private void cacheCrashFile(String fileName) {
SharedPreferences sp = mContext.getSharedPreferences("crash", Context.MODE_PRIVATE);
// sp.edit().putString("CRASH_FILE_NAME", fileName).commit();
sp.edit().putString("TJTREK_CRASH_FILE_NAME", fileName).commit();
}
/**
* 獲取崩潰文件名稱
*
* @return
*/
public File getCrashFile() {
String crashFileName = mContext.getSharedPreferences("crash",
Context.MODE_PRIVATE).getString("TJTREK_CRASH_FILE_NAME", ""); //CRASH_FILE_NAME
return new File(crashFileName);
}
/**
* 保存獲取的 軟件信息蜻展,設(shè)備信息和出錯(cuò)信息保存在SDcard中
*
* @param ex
* @return
*/
private String saveInfoToSD(Throwable ex) {
String fileName = null;
StringBuffer sb = new StringBuffer();
// 1. 手機(jī)信息 + 應(yīng)用信息 --> 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.崩潰的詳細(xì)信息
sb.append(obtainExceptionInfo(ex));
// 保存文件 手機(jī)應(yīng)用的目錄,并沒有拿手機(jī)sdCard目錄邀摆, 6.0 以上需要?jiǎng)討B(tài)申請(qǐng)權(quán)限
if (Environment.getExternalStorageState().equals(
Environment.MEDIA_MOUNTED)) {
File dir = new File(mContext.getFilesDir() + File.separator + "crash"
+ File.separator);
// 先刪除之前的異常信息
if (dir.exists()) {
// 刪除該目錄下的所有子文件
deleteDir(dir);
}
// 再?gòu)男聞?chuàng)建文件夾
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);
}
/**
* 獲取一些簡(jiǎn)單的信息,軟件版本纵顾,手機(jī)版本,型號(hào)等信息存放在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;
}
/**
* 獲取手機(jī)信息 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();
}
/**
* 獲取系統(tǒng)未捕捉的錯(cuò)誤信息
*
* @param throwable
* @return
*/
private String obtainExceptionInfo(Throwable throwable) {
// Java基礎(chǔ) 異常
StringWriter stringWriter = new StringWriter();
PrintWriter printWriter = new PrintWriter(stringWriter);
throwable.printStackTrace(printWriter);
printWriter.close();
return stringWriter.toString();
}
/**
* 遞歸刪除目錄下的所有文件及子目錄下所有文件
*
* @param dir 將要?jiǎng)h除的文件目錄
* @return boolean Returns "true" if all deletions were successful. If a
* deletion fails, the method stops attempting to delete and returns
* "false".
*/
private boolean deleteDir(File dir) {
if (dir.isDirectory()) {
File[] children = dir.listFiles();
// 遞歸刪除目錄中的子目錄下
for (File child : children) {
child.delete();
}
}
// 目錄此時(shí)為空栋盹,可以刪除
return true;
}
}
2.2: 在自己程序中BaseApplication中的onCreate()方法設(shè)置全局異常捕捉類
/**
* @date : 2014/10/18
* @author Jack-Chen
* @function:
* 自定義的Application 1施逾、運(yùn)行的過(guò)程中只有一個(gè)實(shí)例 2、一個(gè)應(yīng)用程序最先執(zhí)行的方法 運(yùn)行在主線程中
* 注意:在AndroidManifest文件中進(jìn)行注冊(cè)
*/
public class BaseApplication extends Application {
@Override
public void onCreate() {
super.onCreate();
//設(shè)置全局異常捕捉類
ExceptionCrashHandler.getmInstance().init(this);
}
}
2.3:直接在MainActivity的initData()初始化數(shù)據(jù)方法中例获,獲取上次崩潰信息汉额,然后打印即可
@Override
public void initData() {
//每次運(yùn)行程序都會(huì)獲取上次的崩潰信息
File crashFile = ExceptionCrashHandler.getmInstance().getCrashFile() ;
if (crashFile.exists()){
//將保存的崩潰日志讀取出來(lái)并打印,就知道崩潰具體信息榨汤,然后處理
try {
InputStreamReader fileReader = new InputStreamReader(new FileInputStream(crashFile)) ;
char[] buffer = new char[1024] ;
int len=0;
while ((len=fileReader.read(buffer))!=-1){
String str = new String(buffer , 0 , len);
Log.d("=====崩潰信息打印====" , str+"--------") ;
}
} catch (Exception e) {
e.printStackTrace();
}
}
}
以上就是全局異常捕獲及使用步驟蠕搜,如若需要,直接拷貝到自己項(xiàng)目中即可使用
如果對(duì)您有幫助收壕,可以關(guān)注我的簡(jiǎn)書讥脐,我會(huì)持續(xù)更新文章遭居,將自己項(xiàng)目中出現(xiàn)的問(wèn)題、bug旬渠、及解決方法都會(huì)上傳簡(jiǎn)書俱萍,我不希望別人去踩我自己踩過(guò)的坑,因?yàn)槲矣X得這種東西沒必要所有人都去犯告丢,希望自己綿薄之力對(duì)您有幫助