Android 的默认崩溃机制是 APP 闪退,然后显示一个【xxx 已停止运行】的对话框或 Toast,而崩溃的详情只有开发者在 Logcat 里才能看到,用户看到发生了这样的情况肯定一头雾水,的确,这样默认的异常处理方式很不友好,容易造成用户流失。我们现在要做的是,程序发生异常时,新开一个 Activity 向用户致歉,输出详细的异常信息,并提供将异常信息提交给开发者的功能。
首先,在 BaseActivity 里封装方法:
public abstract class BaseActivity extends AppCompatActivity {
private static final AppManager MANAGER = AppManager.get();
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
MANAGER.addActivity(this);
} // onCreate()
@Override
protected void onDestroy() {
super.onDestroy();
MANAGER.removeActivity(this);
} // onDestroy()
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()
String getCrashDump() {
return getIntent().getStringExtra("dump");
} // getCrashDump()
} // BaseActivity Abstract Class
// E.O.F
BaseActivity 里用到了两个自定义类,AppManager 和 CrashActivity。后面添加的这两个类请确保和 BaseActivity 在同一包下。
添加 AppManager 类:
class AppManager {
private static final AppManager MANAGER = new AppManager();
private Stack<BaseActivity> mStack;
private AppManager() {
// 将作用域关键字设置为 private 以隐藏该类的构造器。
mStack = new Stack<>();
} // AppManager() (Class Constructor)
static AppManager get() {
return MANAGER;
} // get()
void addActivity(BaseActivity activity) {
mStack.add(activity);
Log.i("AppManager", "[+] Created: " + activity.getClass().getName());
} // addActivity()
void removeActivity(BaseActivity activity) {
mStack.remove(activity);
Log.i("AppManager", "<-> Removed: " + activity.getClass().getName());
} // removeActivity()
void finishAllExcept(Class<?> cls) {
int i, len;
BaseActivity[] activities;
// 结束活动时会调用活动的 onDestroy() 方法,堆栈的内容会实时改变
// 为避免因此引起的引用错误,先将堆栈的内容复制到一个临时数组里
activities = mStack.toArray(new BaseActivity[0]);
len = activities.length;
for (i = 0; i < len; ++i) {
if (activities[i].getClass() != cls) {
// 从数组里引用活动对象并结束,堆栈内容的改变不影响数组
activities[i].finish();
} // if (activities[i].getClass() != cls)
} // for (i = 0; i < len; ++i)
} // finishAllExcept()
void finishAllActivities() {
int i, len;
BaseActivity[] activities;
// 结束活动时会调用活动的 onDestroy() 方法,堆栈的内容会实时改变
// 为避免因此引起的引用错误,先将堆栈的内容复制到一个临时数组里
activities = mStack.toArray(new BaseActivity[0]);
len = activities.length;
for (i = 0; i < len; ++i) {
// 从数组里引用活动对象并结束,堆栈内容的改变不影响数组
activities[i].finish();
} // for (i = 0; i < len; ++i)
} // finishAllActivities()
} // AppManager Class
// E.O.F
新建 CrashActivity 活动。
活动的布局文件 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">
<!-- 请自行设置 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">
程序发生了非预期错误
\n非常抱歉给您造成不便
\n以下是错误详情
</string>
CrashActivity.java 代码:
public class CrashActivity extends BaseActivity { // 注意此处是继承 BaseActivity
@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()
@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()
@Override
protected void onUserLeaveHint() {
AppManager.get().finishAllActivities();
} // onUserLeaveHint()
} // CrashActivity Class
// E.O.F
下面我们要做的就是,在程序抛出异常时捕获它,并将异常内容带入 CrashActivity 中。要实现这样的操作,我们需要在 Activity 中的所有 public 和 protected 方法里添加 try/catch 语句块。(private 方法不用添加,因为 private 方法也必然是由某个 public 或 protected 方法调用的,而调用它的 public/protected 方法已经在抓捕异常了)
我们在 MainActivity 里添加一个按钮。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">崩溃测试</string>
MainActivity.java 代码:
public class MainActivity extends BaseActivity { // 注意此处是继承 BaseActivity
@Override
protected void onCreate(Bundle savedInstanceState) {
// protected 方法必须以 try/catch 包裹
// 在 catch 中加入 crash(e); 语句实现友好崩溃
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); 语句实现友好崩溃
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 包裹
// 除非调用了带 throws 关键字的方法强制要求捕获异常
arr[4] = 4; // 因为传入的 arr 数组长度为 4,所以此处会抛出数组越界异常
} // crashTest()
} // MainActivity Class
// E.O.F
安装到手机上测试一下
点击【崩溃测试】按钮
这里的演示程序并没有添加向开发者提交错误报告的功能,当然本文的重点在于实现友好的崩溃界面,在此基础上的更多功能请读者自行实现。
以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持编程网。