Android實(shí)現(xiàn)友好崩潰界面
Android 的默認(rèn)崩潰機(jī)制是 APP 閃退,然后顯示一個(gè)【xxx 已停止運(yùn)行】的對(duì)話框或 Toast,而崩潰的詳情只有開(kāi)發(fā)者在 Logcat 里才能看到,用戶看到發(fā)生了這樣的情況肯定一頭霧水,的確,這樣默認(rèn)的異常處理方式很不友好,容易造成用戶流失。我們現(xiàn)在要做的是,程序發(fā)生異常時(shí),新開(kāi)一個(gè) Activity 向用戶致歉,輸出詳細(xì)的異常信息,并提供將異常信息提交給開(kāi)發(fā)者的功能。
首先,在 BaseActivity 里封裝方法:
/** * BaseActivity: 該抽象類(lèi)定義所有活動(dòng)均擁有的共同屬性。 * 本 APP 中所有活動(dòng)對(duì)象均繼承此類(lèi)。 */ public abstract class BaseActivity extends AppCompatActivity { private static final AppManager MANAGER = AppManager.get(); /** * onCreate(): 重寫(xiě)父類(lèi)的 onCreate() 方法,向應(yīng)用管理器中添加本活動(dòng)。 */ @Override protected void onCreate(Bundle savedInstanceState) { super.onCreate(savedInstanceState); MANAGER.addActivity(this); } // onCreate() /** * onDestroy(): 重寫(xiě)父類(lèi)的 onDestroy() 方法,從應(yīng)用管理器中移除本活動(dòng)。 */ @Override protected void onDestroy() { super.onDestroy(); MANAGER.removeActivity(this); } // onDestroy() /** * crash(): 捕獲到非預(yù)期的異常后強(qiáng)制令程序崩潰。 * * @param e 傳入造成崩潰的異常對(duì)象。 */ protected void crash(Exception e) { Intent i; String dump; PrintWriter pw; StringWriter sw; sw = new StringWriter(); pw = new PrintWriter(sw); e.printStackTrace(pw); pw.flush(); dump = sw.toString(); i = new Intent(this, CrashActivity.class); i.putExtra("dump", dump); startActivity(i); MANAGER.finishAllExcept(CrashActivity.class); } // crash() /** * getCrashDump(): 僅限 CrashActivity 調(diào)用。 * 獲得傳入的 dump 信息。 * * @return 傳入的 dump 信息。 */ String getCrashDump() { return getIntent().getStringExtra("dump"); } // getCrashDump() } // BaseActivity Abstract Class // E.O.F
BaseActivity 里用到了兩個(gè)自定義類(lèi),AppManager 和 CrashActivity。后面添加的這兩個(gè)類(lèi)請(qǐng)確保和 BaseActivity 在同一包下。
添加 AppManager 類(lèi):
/** * AppManager: 用于對(duì)活動(dòng)進(jìn)行管理。該模塊僅限 base 包內(nèi)使用。 * 該模塊為單一實(shí)例,您需要調(diào)用 AppManager.get() 獲取實(shí)例后再調(diào)用方法。 * <p> * 為確保應(yīng)用管理器正常工作,請(qǐng)新建一個(gè)繼承 Activity 的抽象類(lèi) BaseActivity, * 然后重寫(xiě) BaseActivity 類(lèi)的 onCreate() 和 onDestroy() 方法。 * 請(qǐng)給 BaseActivity 類(lèi)的 onCreate() 方法添加如下代碼: * AppManager.get().addActivity(this); * 請(qǐng)給 BaseActivity 類(lèi)的 onDestroy() 方法添加如下代碼: * AppManager.get().removeActivity(this); * 最后,確保本 APP 內(nèi)的所有活動(dòng)類(lèi)均繼承于 BaseActivity 類(lèi)。 */ class AppManager { private static final AppManager MANAGER = new AppManager(); private Stack<BaseActivity> mStack; private AppManager() { // 將作用域關(guān)鍵字設(shè)置為 private 以隱藏該類(lèi)的構(gòu)造器。 mStack = new Stack<>(); } // AppManager() (Class Constructor) /** * get(): 獲得 AppManager 類(lèi)的單例。 * * @return 該類(lèi)的單例 MANAGER。 */ static AppManager get() { return MANAGER; } // get() /** * addActivity(): 向堆棧中添加一個(gè)活動(dòng)對(duì)象。 * * @param activity 要添加的活動(dòng)對(duì)象。 */ void addActivity(BaseActivity activity) { mStack.add(activity); Log.i("AppManager", "[+] Created: " + activity.getClass().getName()); } // addActivity() /** * removeActivity(): 從堆棧中移除一個(gè)活動(dòng)對(duì)象。 * * @param activity 要移除的活動(dòng)對(duì)象。 */ void removeActivity(BaseActivity activity) { mStack.remove(activity); Log.i("AppManager", "<-> Removed: " + activity.getClass().getName()); } // removeActivity() /** * finishAllExcept(): 除一個(gè)特定活動(dòng)外,結(jié)束堆棧中其余所有活動(dòng)。 * 結(jié)束活動(dòng)時(shí)會(huì)觸發(fā) BaseActivity 類(lèi)的 onDestroy()方法, * 堆棧中的活動(dòng)對(duì)象會(huì)同步移除。 * * @param cls 要保留的活動(dòng)的類(lèi)名(xxxActivity.class) */ void finishAllExcept(Class<?> cls) { int i, len; BaseActivity[] activities; // 結(jié)束活動(dòng)時(shí)會(huì)調(diào)用活動(dòng)的 onDestroy() 方法,堆棧的內(nèi)容會(huì)實(shí)時(shí)改變 // 為避免因此引起的引用錯(cuò)誤,先將堆棧的內(nèi)容復(fù)制到一個(gè)臨時(shí)數(shù)組里 activities = mStack.toArray(new BaseActivity[0]); len = activities.length; for (i = 0; i < len; ++i) { if (activities[i].getClass() != cls) { // 從數(shù)組里引用活動(dòng)對(duì)象并結(jié)束,堆棧內(nèi)容的改變不影響數(shù)組 activities[i].finish(); } // if (activities[i].getClass() != cls) } // for (i = 0; i < len; ++i) } // finishAllExcept() /** * finishAllActivities(): 結(jié)束堆棧中的所有活動(dòng)。 * 結(jié)束活動(dòng)時(shí)會(huì)觸發(fā) BaseActivity 類(lèi)的 onDestroy()方法, * 堆棧中的活動(dòng)對(duì)象會(huì)同步移除。 */ void finishAllActivities() { int i, len; BaseActivity[] activities; // 結(jié)束活動(dòng)時(shí)會(huì)調(diào)用活動(dòng)的 onDestroy() 方法,堆棧的內(nèi)容會(huì)實(shí)時(shí)改變 // 為避免因此引起的引用錯(cuò)誤,先將堆棧的內(nèi)容復(fù)制到一個(gè)臨時(shí)數(shù)組里 activities = mStack.toArray(new BaseActivity[0]); len = activities.length; for (i = 0; i < len; ++i) { // 從數(shù)組里引用活動(dòng)對(duì)象并結(jié)束,堆棧內(nèi)容的改變不影響數(shù)組 activities[i].finish(); } // for (i = 0; i < len; ++i) } // finishAllActivities() } // AppManager Class // E.O.F
新建 CrashActivity 活動(dòng)。
活動(dòng)的布局文件 activity_crash.xml
<?xml version="1.0" encoding="utf-8"?> <android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" android:background="#1A237E" tools:context=".base.CrashActivity"> <!-- 請(qǐng)自行設(shè)置 background 和 textColor --> <TextView android:id="@+id/lblCrashMsg" android:layout_width="wrap_content" android:layout_height="wrap_content" android:layout_marginStart="8dp" android:layout_marginLeft="8dp" android:layout_marginTop="8dp" android:layout_marginEnd="8dp" android:layout_marginRight="8dp" android:layout_marginBottom="8dp" android:gravity="center" android:text="@string/lblCrashMsg" android:textAppearance="?android:attr/textAppearanceMedium" android:textColor="#EEEEEE" app:layout_constraintBottom_toTopOf="@+id/lblCrashDetail" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintStart_toStartOf="parent" app:layout_constraintTop_toTopOf="parent" /> <TextView android:id="@+id/lblCrashDetail" android:layout_width="0dp" android:layout_height="0dp" android:layout_marginStart="8dp" android:layout_marginLeft="8dp" android:layout_marginTop="8dp" android:layout_marginEnd="8dp" android:layout_marginRight="8dp" android:layout_marginBottom="8dp" android:textColor="#EEEEEE" android:typeface="monospace" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintEnd_toEndOf="parent" app:layout_constraintStart_toStartOf="parent" app:layout_constraintTop_toBottomOf="@+id/lblCrashMsg" /> </android.support.constraint.ConstraintLayout>
字符串資源 strings.xml 里添加
<string name="lblCrashMsg"> 程序發(fā)生了非預(yù)期錯(cuò)誤 \n非常抱歉給您造成不便 \n以下是錯(cuò)誤詳情 </string>
CrashActivity.java 代碼:
/** * CrashActivity: 該活動(dòng)由任意活動(dòng)調(diào)用 crash() 方法激活。輸出拋出的異常信息。 */ public class CrashActivity extends BaseActivity { // 注意此處是繼承 BaseActivity /** * onCreate(): 活動(dòng)創(chuàng)建時(shí)觸發(fā)。 */ @Override protected void onCreate(Bundle savedInstanceState) { String dump; TextView lblDetail; super.onCreate(savedInstanceState); setContentView(R.layout.activity_crash); dump = getCrashDump(); lblDetail = findViewById(R.id.lblCrashDetail); lblDetail.setText(dump); lblDetail.setMovementMethod(ScrollingMovementMethod.getInstance()); } // onCreate() /** * onKeyDown(): 按下回退鍵時(shí)觸發(fā)。 * 直接退出程序。 */ @Override public boolean onKeyDown(int keyCode, KeyEvent event) { if (keyCode == KeyEvent.KEYCODE_BACK) { AppManager.get().finishAllActivities(); return true; } // if (keyCode == KeyEvent.KEYCODE_BACK) else { return super.onKeyDown(keyCode, event); } // else } // onKeyDown() /** * onUserLeaveHint(): 按下 HOME 鍵退回桌面時(shí)觸發(fā)。直接退出程序。 */ @Override protected void onUserLeaveHint() { AppManager.get().finishAllActivities(); } // onUserLeaveHint() } // CrashActivity Class // E.O.F
下面我們要做的就是,在程序拋出異常時(shí)捕獲它,并將異常內(nèi)容帶入 CrashActivity 中。要實(shí)現(xiàn)這樣的操作,我們需要在 Activity 中的所有 public 和 protected 方法里添加 try/catch 語(yǔ)句塊。(private 方法不用添加,因?yàn)?private 方法也必然是由某個(gè) public 或 protected 方法調(diào)用的,而調(diào)用它的 public/protected 方法已經(jīng)在抓捕異常了)
我們?cè)?MainActivity 里添加一個(gè)按鈕。activity_main.xml 布局代碼如下:
<?xml version="1.0" encoding="utf-8"?> <android.support.constraint.ConstraintLayout xmlns:android="http://schemas.android.com/apk/res/android" xmlns:app="http://schemas.android.com/apk/res-auto" xmlns:tools="http://schemas.android.com/tools" android:layout_width="match_parent" android:layout_height="match_parent" tools:context=".MainActivity"> <Button android:layout_width="wrap_content" android:layout_height="wrap_content" android:onClick="onBtnCrashTestTapped" android:text="@string/btnMainCrashTest" app:layout_constraintBottom_toBottomOf="parent" app:layout_constraintLeft_toLeftOf="parent" app:layout_constraintRight_toRightOf="parent" app:layout_constraintTop_toTopOf="parent" /> </android.support.constraint.ConstraintLayout>
strings.xml 里添加:
<string name="btnMainCrashTest">崩潰測(cè)試</string>
MainActivity.java 代碼:
public class MainActivity extends BaseActivity { // 注意此處是繼承 BaseActivity @Override protected void onCreate(Bundle savedInstanceState) { // protected 方法必須以 try/catch 包裹 // 在 catch 中加入 crash(e); 語(yǔ)句實(shí)現(xiàn)友好崩潰 try { super.onCreate(savedInstanceState); setContentView(R.layout.activity_main); } // try catch (Exception e) { crash(e); } // catch (Exception e) } // onCreate() public void onBtnCrashTestTapped(View v) { int[] arr; // public 方法必須以 try/catch 包裹 // 在 catch 中加入 crash(e); 語(yǔ)句實(shí)現(xiàn)友好崩潰 try { arr = new int[4]; crashTest(arr); } // try catch (Exception e) { crash(e); } // catch (Exception e) } // onBtnCrashTestTapped() private void crashTest(int[] arr) { // private 方法不用以 try/catch 包裹 // 除非調(diào)用了帶 throws 關(guān)鍵字的方法強(qiáng)制要求捕獲異常 arr[4] = 4; // 因?yàn)閭魅氲?arr 數(shù)組長(zhǎng)度為 4,所以此處會(huì)拋出數(shù)組越界異常 } // crashTest() } // MainActivity Class // E.O.F
安裝到手機(jī)上測(cè)試一下
點(diǎn)擊【崩潰測(cè)試】按鈕
這里的演示程序并沒(méi)有添加向開(kāi)發(fā)者提交錯(cuò)誤報(bào)告的功能,當(dāng)然本文的重點(diǎn)在于實(shí)現(xiàn)友好的崩潰界面,在此基礎(chǔ)上的更多功能請(qǐng)讀者自行實(shí)現(xiàn)。
以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
相關(guān)文章
Android實(shí)現(xiàn)擴(kuò)展Menu的方法
這篇文章主要介紹了Android實(shí)現(xiàn)擴(kuò)展Menu的方法,涉及Android操作menu菜單的相關(guān)技巧,具有一定參考借鑒價(jià)值,需要的朋友可以參考下2015-10-10Android Drawerlayout側(cè)拉欄事件傳遞問(wèn)題的解決方法
這篇文章主要為大家詳細(xì)介紹了Android Drawerlayout側(cè)拉欄事件傳遞問(wèn)題的解決方法,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下2016-11-11超簡(jiǎn)單Android集成華為HMS Scankit 掃碼SDK實(shí)現(xiàn)掃一掃二維碼
這篇文章主要介紹了超簡(jiǎn)單Android集成華為HMS Scankit 掃碼SDK實(shí)現(xiàn)掃一掃二維碼,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧2020-03-03android中打開(kāi)相機(jī)、打開(kāi)相冊(cè)進(jìn)行圖片的獲取示例
本篇文章主要介紹了android中打開(kāi)相機(jī)、打開(kāi)相冊(cè)進(jìn)行圖片的獲取示例,非常具有實(shí)用價(jià)值,需要的朋友可以參考下。2017-01-01Android中通知欄跳動(dòng)問(wèn)題解決方法
這篇文章主要介紹了Android中通知欄跳動(dòng)問(wèn)題解決方法,導(dǎo)致這個(gè)問(wèn)題的原因是when這個(gè)屬性值,默認(rèn)它是使用的系統(tǒng)當(dāng)前時(shí)間,這就是導(dǎo)致跳動(dòng)問(wèn)題的原因,指定一個(gè)固定時(shí)間即可解決這個(gè)問(wèn)題,需要的朋友可以參考下2015-01-01Android SDK Manager國(guó)內(nèi)無(wú)法更新的解決方案
本文主要介紹Android SDK Manager國(guó)內(nèi)無(wú)法更新的解決方案,這里提供了解決方法,及簡(jiǎn)單說(shuō)明實(shí)現(xiàn)流程,有興趣的小伙伴可以參考下2016-09-09Android?通過(guò)productFlavors實(shí)現(xiàn)多渠道打包方法示例
這篇文章主要為大家介紹了Android?通過(guò)productFlavors實(shí)現(xiàn)多渠道打包方法示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-02-02Android手機(jī)通過(guò)rtp發(fā)送aac數(shù)據(jù)給vlc播放的實(shí)現(xiàn)步驟
這篇文章主要介紹了Android手機(jī)通過(guò)rtp發(fā)送aac數(shù)據(jù)給vlc播放的實(shí)現(xiàn)步驟,幫助大家更好的理解和學(xué)習(xí)使用Android,感興趣的朋友可以了解下2021-04-04