文章详情

短信预约-IT技能 免费直播动态提醒

请输入下面的图形验证码

提交验证

短信预约提醒成功

Android怎样实现友好崩溃界面

2023-06-21 20:15

关注

Android怎样实现友好崩溃界面,相信很多没有经验的人对此束手无策,为此本文总结了问题出现的原因和解决方法,通过这篇文章希望你能解决这个问题。

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

安装到手机上测试一下

Android怎样实现友好崩溃界面

点击【崩溃测试】按钮

Android怎样实现友好崩溃界面

这里的演示程序并没有添加向开发者提交错误报告的功能,当然重点在于实现友好的崩溃界面。

看完上述内容,你们掌握Android怎样实现友好崩溃界面的方法了吗?如果还想学到更多技能或想了解更多相关内容,欢迎关注编程网行业资讯频道,感谢各位的阅读!

阅读原文内容投诉

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

软考中级精品资料免费领

  • 历年真题答案解析
  • 备考技巧名师总结
  • 高频考点精准押题
  • 2024年上半年信息系统项目管理师第二批次真题及答案解析(完整版)

    难度     813人已做
    查看
  • 【考后总结】2024年5月26日信息系统项目管理师第2批次考情分析

    难度     354人已做
    查看
  • 【考后总结】2024年5月25日信息系统项目管理师第1批次考情分析

    难度     318人已做
    查看
  • 2024年上半年软考高项第一、二批次真题考点汇总(完整版)

    难度     435人已做
    查看
  • 2024年上半年系统架构设计师考试综合知识真题

    难度     224人已做
    查看

相关文章

发现更多好内容

猜你喜欢

AI推送时光机
位置:首页-资讯-后端开发
咦!没有更多了?去看看其它编程学习网 内容吧
首页课程
资料下载
问答资讯