【Android开发经验】我们要友好的告诉用户,程序要崩溃了

2024-05-13 17:32

本文主要是介绍【Android开发经验】我们要友好的告诉用户,程序要崩溃了,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

原帖地址:http://blog.csdn.net/zhaokaiqiang1992


虽然我们的程序在正式上线之前,都会经过严格的测试,从而保证程序的健壮性和良好的用户体验,但是,一个人的测试或者是几个人的测试团队,都不能和上万甚至数十万的用户相比。因此,前期刚上线的程序在用户手里被玩崩了,也是很常见的事,但是,如果我们不做特殊处理,系统自带的程序崩溃提示真的太吓人了,用户看到之后会不知所措,因此,我们需要一个解决方案,就是在程序即将崩溃的时候,给用户一个更加友好的提示,来告诉他,程序马上要崩溃了。

    就像是下面这样:



     这样是不是更加友好一点呢?

    下面讲解如何实现。

     如果要实现这种功能,我们需要关注的是Thread类里面的一个接口,UncaughtExceptionHandler,还有一个设置Thread.setDefaultUncaughtExceptionHandler(),这两个东西到底是干嘛的呢?

    UncaughtExceptionHandler 这个接口是当Thread因为未被捕获的异常而要被终止的时候,会被调用的回调接口。

    而Thread.setDefaultUncaughtExceptionHandler()这个方法,则是设置当线程由于未捕获到异常而突然终止,并且没有为该线程定义其他处理程序时所调用的默认处理程序。

    因此,如果我们想自己处理程序要崩溃时的处理逻辑,我们只需要实现UncaughtExceptionHandler接口,并调用Thread.setDefaultUncaughtExceptionHandler()设置即可。

    下面是示例代码

    首先,先给出上面效果图中的Activity的代码

[java]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. package com.example.exceptiondemo;  
  2.   
  3. import android.app.Activity;  
  4. import android.os.Bundle;  
  5. import android.view.View;  
  6.   
  7. public class MainActivity extends Activity {  
  8.   
  9.     @Override  
  10.     protected void onCreate(Bundle savedInstanceState) {  
  11.         super.onCreate(savedInstanceState);  
  12.         setContentView(R.layout.activity_main);  
  13.         // 注册默认的未捕捉异常处理类  
  14.         Thread.setDefaultUncaughtExceptionHandler(AppException  
  15.                 .getAppExceptionHandler());  
  16.         AppManager.getAppManager().addActivity(this);  
  17.   
  18.     }  
  19.   
  20.     public void btn(View view) {  
  21.         // 除零错误,程序会崩溃  
  22.         int c = 1 / 0;  
  23.     }  
  24.   
  25. }  

     在这段代码里面,我们故意写了一个会出现异常的操作,除零,因此只要点击按钮,程序就会崩溃。

    下面,我们要自己实现接口,这里,我继承了Exception类。

[java]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. package com.example.exceptiondemo;  
  2.   
  3. import java.lang.Thread.UncaughtExceptionHandler;  
  4.   
  5. import android.app.Activity;  
  6. import android.app.AlertDialog;  
  7. import android.content.DialogInterface;  
  8. import android.content.DialogInterface.OnClickListener;  
  9. import android.os.Looper;  
  10. import android.widget.Toast;  
  11.   
  12. /** 
  13.  *  
  14.  * @ClassName: com.example.exceptiondemo.AppException 
  15.  * @Description: 应用程序异常类:用于捕获异常 
  16.  * @author zhaokaiqiang 
  17.  * @date 2014-11-2 下午10:06:49 
  18.  *  
  19.  */  
  20.   
  21. public class AppException extends Exception implements UncaughtExceptionHandler {  
  22.   
  23.     private static final long serialVersionUID = -6262909398048670705L;  
  24.   
  25.     private String message;  
  26.   
  27.     private Thread.UncaughtExceptionHandler mDefaultHandler;  
  28.   
  29.     private AppException() {  
  30.         super();  
  31.         this.mDefaultHandler = Thread.getDefaultUncaughtExceptionHandler();  
  32.     }  
  33.   
  34.     public AppException(String message, Exception excp) {  
  35.         super(message, excp);  
  36.         this.mDefaultHandler = Thread.getDefaultUncaughtExceptionHandler();  
  37.     }  
  38.   
  39.     public String getMessage() {  
  40.         return message;  
  41.     }  
  42.   
  43.     public void setMessage(String message) {  
  44.         this.message = message;  
  45.     }  
  46.   
  47.     /** 
  48.      * 获取APP异常崩溃处理对象 
  49.      *  
  50.      * @param context 
  51.      * @return 
  52.      */  
  53.     public static AppException getAppExceptionHandler() {  
  54.         return new AppException();  
  55.     }  
  56.   
  57.     @Override  
  58.     public void uncaughtException(Thread thread, Throwable ex) {  
  59.   
  60.         if (!handleException(ex) && mDefaultHandler != null) {  
  61.             mDefaultHandler.uncaughtException(thread, ex);  
  62.         }  
  63.   
  64.     }  
  65.   
  66.     /** 
  67.      * 自定义异常处理 
  68.      *  
  69.      * @param ex 
  70.      * @return true:处理了该异常信息;否则返回false 
  71.      */  
  72.     private boolean handleException(Throwable ex) {  
  73.         if (ex == null) {  
  74.             return false;  
  75.         }  
  76.   
  77.         final Activity activity = AppManager.getAppManager().currentActivity();  
  78.   
  79.         if (activity == null) {  
  80.             return false;  
  81.         }  
  82.   
  83.         new Thread() {  
  84.             @Override  
  85.             public void run() {  
  86.                 Looper.prepare();  
  87.                 Toast.makeText(activity, "程序要崩了", Toast.LENGTH_SHORT).show();  
  88.                 new AlertDialog.Builder(activity).setTitle("提示")  
  89.                         .setCancelable(false).setMessage("亲,程序马上崩溃了...")  
  90.                         .setNeutralButton("没关系"new OnClickListener() {  
  91.                             @Override  
  92.                             public void onClick(DialogInterface dialog,  
  93.                                     int which) {  
  94.                                 AppManager.getAppManager().exitApp(activity);  
  95.                             }  
  96.                         }).create().show();  
  97.                 Looper.loop();  
  98.             }  
  99.         }.start();  
  100.   
  101.         return true;  
  102.     }  
  103.   
  104. }  

     实现借口之后,我们需要在uncaughtException()方法里面处理自己的逻辑,我在这里面弹出对话框提示用户程序崩溃,这样,就比较友好一些。在代码里面的AppManager类是一个Activity的管理类,下面是代码

[java]  view plain copy 在CODE上查看代码片 派生到我的代码片
  1. package com.example.exceptiondemo;  
  2.   
  3. import java.util.Stack;  
  4.   
  5. import android.app.Activity;  
  6. import android.app.ActivityManager;  
  7. import android.content.Context;  
  8.   
  9. /** 
  10.  *  
  11.  * @ClassName: net.oschina.app.AppManager 
  12.  * @Description: Activity管理类:用于管理Activity和退出程序 
  13.  * @author zhaokaiqiang 
  14.  * @date 2014-11-2 上午11:27:55 
  15.  *  
  16.  */  
  17. public class AppManager {  
  18.   
  19.     private static Stack<Activity> activityStack;  
  20.     private static AppManager instance;  
  21.   
  22.     private AppManager() {  
  23.     }  
  24.   
  25.     /** 
  26.      * 单一实例 
  27.      */  
  28.     public static AppManager getAppManager() {  
  29.         if (instance == null) {  
  30.             instance = new AppManager();  
  31.         }  
  32.         return instance;  
  33.     }  
  34.   
  35.     /** 
  36.      * 添加Activity到堆栈 
  37.      */  
  38.     public void addActivity(Activity activity) {  
  39.         if (activityStack == null) {  
  40.             activityStack = new Stack<Activity>();  
  41.         }  
  42.         activityStack.add(activity);  
  43.     }  
  44.   
  45.     /** 
  46.      * 获取当前Activity(堆栈中最后一个压入的) 
  47.      */  
  48.     public Activity currentActivity() {  
  49.         Activity activity = activityStack.lastElement();  
  50.         return activity;  
  51.     }  
  52.   
  53.     /** 
  54.      * 结束当前Activity(堆栈中最后一个压入的) 
  55.      */  
  56.     public void finishActivity() {  
  57.         Activity activity = activityStack.lastElement();  
  58.         finishActivity(activity);  
  59.     }  
  60.   
  61.     /** 
  62.      * 结束指定的Activity 
  63.      */  
  64.     public void finishActivity(Activity activity) {  
  65.         if (activity != null) {  
  66.             activityStack.remove(activity);  
  67.             activity.finish();  
  68.             activity = null;  
  69.         }  
  70.     }  
  71.   
  72.     /** 
  73.      * 结束指定类名的Activity 
  74.      */  
  75.     public void finishActivity(Class<?> cls) {  
  76.         for (Activity activity : activityStack) {  
  77.             if (activity.getClass().equals(cls)) {  
  78.                 finishActivity(activity);  
  79.             }  
  80.         }  
  81.     }  
  82.   
  83.     /** 
  84.      * 结束所有Activity 
  85.      */  
  86.     public void finishAllActivity() {  
  87.         for (int i = 0, size = activityStack.size(); i < size; i++) {  
  88.             if (null != activityStack.get(i)) {  
  89.                 activityStack.get(i).finish();  
  90.             }  
  91.         }  
  92.         activityStack.clear();  
  93.     }  
  94.   
  95.     /** 
  96.      * 退出应用程序 
  97.      */  
  98.     public void exitApp(Context context) {  
  99.         try {  
  100.             finishAllActivity();  
  101.             ActivityManager activityMgr = (ActivityManager) context  
  102.                     .getSystemService(Context.ACTIVITY_SERVICE);  
  103.             activityMgr.killBackgroundProcesses(context.getPackageName());  
  104.             System.exit(0);  
  105.         } catch (Exception e) {  
  106.         }  
  107.     }  
  108. }  

    Demo下载地址:https://github.com/ZhaoKaiQiang/ExceptionDemo


自我感觉放在application里面会更好一些..

package com.example.testuncaultexception;import android.app.Application;
import android.content.Context;public class MyApp extends Application {@Overridepublic void onCreate() {super.onCreate();Thread.setDefaultUncaughtExceptionHandler(AppException.getAppExceptionHandler());}}


这篇关于【Android开发经验】我们要友好的告诉用户,程序要崩溃了的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/986360

相关文章

这15个Vue指令,让你的项目开发爽到爆

1. V-Hotkey 仓库地址: github.com/Dafrok/v-ho… Demo: 戳这里 https://dafrok.github.io/v-hotkey 安装: npm install --save v-hotkey 这个指令可以给组件绑定一个或多个快捷键。你想要通过按下 Escape 键后隐藏某个组件,按住 Control 和回车键再显示它吗?小菜一碟: <template

Hadoop企业开发案例调优场景

需求 (1)需求:从1G数据中,统计每个单词出现次数。服务器3台,每台配置4G内存,4核CPU,4线程。 (2)需求分析: 1G / 128m = 8个MapTask;1个ReduceTask;1个mrAppMaster 平均每个节点运行10个 / 3台 ≈ 3个任务(4    3    3) HDFS参数调优 (1)修改:hadoop-env.sh export HDFS_NAMENOD

嵌入式QT开发:构建高效智能的嵌入式系统

摘要: 本文深入探讨了嵌入式 QT 相关的各个方面。从 QT 框架的基础架构和核心概念出发,详细阐述了其在嵌入式环境中的优势与特点。文中分析了嵌入式 QT 的开发环境搭建过程,包括交叉编译工具链的配置等关键步骤。进一步探讨了嵌入式 QT 的界面设计与开发,涵盖了从基本控件的使用到复杂界面布局的构建。同时也深入研究了信号与槽机制在嵌入式系统中的应用,以及嵌入式 QT 与硬件设备的交互,包括输入输出设

OpenHarmony鸿蒙开发( Beta5.0)无感配网详解

1、简介 无感配网是指在设备联网过程中无需输入热点相关账号信息,即可快速实现设备配网,是一种兼顾高效性、可靠性和安全性的配网方式。 2、配网原理 2.1 通信原理 手机和智能设备之间的信息传递,利用特有的NAN协议实现。利用手机和智能设备之间的WiFi 感知订阅、发布能力,实现了数字管家应用和设备之间的发现。在完成设备间的认证和响应后,即可发送相关配网数据。同时还支持与常规Sof

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟&nbsp;开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚&nbsp;第一站:海量资源,应有尽有 走进“智听

活用c4d官方开发文档查询代码

当你问AI助手比如豆包,如何用python禁止掉xpresso标签时候,它会提示到 这时候要用到两个东西。https://developers.maxon.net/论坛搜索和开发文档 比如这里我就在官方找到正确的id描述 然后我就把参数标签换过来

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

Android平台播放RTSP流的几种方案探究(VLC VS ExoPlayer VS SmartPlayer)

技术背景 好多开发者需要遴选Android平台RTSP直播播放器的时候,不知道如何选的好,本文针对常用的方案,做个大概的说明: 1. 使用VLC for Android VLC Media Player(VLC多媒体播放器),最初命名为VideoLAN客户端,是VideoLAN品牌产品,是VideoLAN计划的多媒体播放器。它支持众多音频与视频解码器及文件格式,并支持DVD影音光盘,VCD影

Linux_kernel驱动开发11

一、改回nfs方式挂载根文件系统         在产品将要上线之前,需要制作不同类型格式的根文件系统         在产品研发阶段,我们还是需要使用nfs的方式挂载根文件系统         优点:可以直接在上位机中修改文件系统内容,延长EMMC的寿命         【1】重启上位机nfs服务         sudo service nfs-kernel-server resta

【区块链 + 人才服务】区块链集成开发平台 | FISCO BCOS应用案例

随着区块链技术的快速发展,越来越多的企业开始将其应用于实际业务中。然而,区块链技术的专业性使得其集成开发成为一项挑战。针对此,广东中创智慧科技有限公司基于国产开源联盟链 FISCO BCOS 推出了区块链集成开发平台。该平台基于区块链技术,提供一套全面的区块链开发工具和开发环境,支持开发者快速开发和部署区块链应用。此外,该平台还可以提供一套全面的区块链开发教程和文档,帮助开发者快速上手区块链开发。