Android異常處理最佳實(shí)踐
一個(gè)好的app 異常處理機(jī)制 我認(rèn)為應(yīng)該至少包含以下幾個(gè)功能:
1.能把錯誤信息上傳到服務(wù)器 讓開發(fā)者可以持續(xù)改進(jìn)app
2.錯誤信息至少應(yīng)該包含 是否在主進(jìn)程 是否在主線程 等可以幫助程序員定位的信息
3.最好包含手機(jī)硬件及軟件信息。
4.主進(jìn)程引發(fā)的異常 最好交由系統(tǒng)自己處理 也就是讓用戶可以感知到 那種(當(dāng)然你也可以自己定義一套更有意思的感知系統(tǒng)對話框等,具體可參考各種有意思的404界面)
5.子進(jìn)程引發(fā)的異常最好別讓用戶感知到。比如push之類的 這種 和用戶感知弱關(guān)聯(lián)的這種。最好發(fā)生異常就直接kill掉即可。不要再交給系統(tǒng)處理。
下面上代碼。
package com.example.administrator.exceptiontest; import android.app.ActivityManager; import android.app.Application; import android.content.Context; /** * Created by Administrator on 2015/12/9. */ public class BaseApplication extends Application { public static Context mContext; //默認(rèn)的異常處理 public static Thread.UncaughtExceptionHandler defaultUncaughtExceptionHandler; @Override public void onCreate() { super.onCreate(); mContext = this; //先取得默認(rèn)的異常處理handler defaultUncaughtExceptionHandler = Thread.getDefaultUncaughtExceptionHandler(); Thread.setDefaultUncaughtExceptionHandler(new BaseUncaughtExceptionHandler()); } }
package com.example.administrator.exceptiontest; import android.app.ActivityManager; import android.content.Context; import android.os.Looper; /** * Created by Administrator on 2015/12/9. */ public class Utils { /** * 判斷是否在主線程中執(zhí)行 如果是返回true 不是返回false */ public static boolean isInMainThread() { //注意這個(gè)地方 我們無法一定可以取得myLooper的值的 比如說 你的thread 沒有綁定消息循環(huán) //那你的mylooper就返回的一定是null了,只有綁定了以后才會返回相應(yīng)的值 return Looper.myLooper() == Looper.getMainLooper(); } //判斷是否是主進(jìn)程 如果是 就返回true 否則返回false public static boolean isMainProcess(Context context) { return context.getPackageName().equals(getProcessName(context)); } //取得進(jìn)程名 public static String getProcessName(Context context) { String currentProcessName = ""; int pid = android.os.Process.myPid(); ActivityManager manager = (ActivityManager) context.getSystemService(Context.ACTIVITY_SERVICE); for (ActivityManager.RunningAppProcessInfo processInfo : manager.getRunningAppProcesses()) { if (processInfo.pid == pid) { currentProcessName = processInfo.processName; break; } } return currentProcessName; } }
package com.example.administrator.exceptiontest; import java.io.PrintWriter; import java.io.StringWriter; import java.io.Writer; /** * Created by Administrator on 2015/12/9. */ public class BaseUncaughtExceptionHandler implements Thread.UncaughtExceptionHandler { @Override public void uncaughtException(Thread thread, Throwable ex) { Writer resultWriter = new StringWriter(); PrintWriter printWriter = new PrintWriter(resultWriter); ex.printStackTrace(printWriter); StringBuffer sb = new StringBuffer(); sb.append("是否在主線程中發(fā)生異常" + Utils.isInMainThread() + "\n"); sb.append("是否在主進(jìn)程中發(fā)生異常" + Utils.isMainProcess(BaseApplication.mContext) + "\n"); String errorReport = sb.toString() + resultWriter.toString(); //這個(gè)地方最好把收集到的錯誤信息日志 上傳至服務(wù)器 便于開發(fā)者定位修改問題。 //如果是主進(jìn)程發(fā)生了異常 那還是交給系統(tǒng)自己默認(rèn)的異常處理來做。讓用戶感知到,否則用戶啥都不知道 體驗(yàn)并不好 //你當(dāng)然也可以自己定義特殊的錯誤提示 比如一些有意思的dialog什么的 if (Utils.isMainProcess(BaseApplication.mContext)) { BaseApplication.defaultUncaughtExceptionHandler.uncaughtException(thread, ex); } else { //如果是子進(jìn)程發(fā)生異常 不要給提示 直接殺掉子進(jìn)程最好 別讓用戶感知到 android.os.Process.killProcess(android.os.Process.myPid()); } } }
以上就是本文的全部內(nèi)容,希望對大家學(xué)習(xí)Android軟件編程有所幫助。
相關(guān)文章
Android自定義ViewGroup實(shí)現(xiàn)可滾動的橫向布局(2)
這篇文章主要介紹了Android自定義ViewGroup實(shí)現(xiàn)可滾動的橫向布局,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-12-12Android Walker登錄記住密碼頁面功能實(shí)現(xiàn)
這篇文章主要為大家詳細(xì)介紹了Android Walker登錄記住密碼頁面功能的實(shí)現(xiàn),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2019-05-05Android利用廣播接收器實(shí)現(xiàn)自動填充短信驗(yàn)證碼
這篇文章主要為大家詳細(xì)介紹了Android利用廣播接收器實(shí)現(xiàn)自動填充短信驗(yàn)證碼,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2017-12-12android listview 水平滾動和垂直滾動的小例子
android listview 水平滾動和垂直滾動的小例子,需要的朋友可以參考一下2013-05-05解析Android中使用自定義字體的實(shí)現(xiàn)方法
本篇文章是對在Android中使用自定義字體的方法進(jìn)行了詳細(xì)的分析介紹。需要的朋友參考下2013-05-05Android Compose實(shí)現(xiàn)伸縮ToolBar的思路詳解
這篇文章主要介紹了Android Compose之伸縮ToolBar的實(shí)現(xiàn),本文給大家分享主要實(shí)現(xiàn)思路及實(shí)現(xiàn)過程,通過實(shí)例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下2021-10-10