有时候由于测试不充分或者程序潜在的问题而导致程序异常崩溃,这个是令人无法接受的,在android中怎样捕获程序的异常崩溃,然后进行一些必要的处理或重新启动

应用这个问题困恼了我很久,今天终于解决了该问题,写篇文章记录一下。

首先捕获程序崩溃的异常就必须了解一下java中UncaughtExceptionHandler这个接口,android沿用了此接口,在android API中:

通过实现此接口,能够处理线程被一个无法捕捉的异常所终止的情况。如上所述的情况,handler将会报告线程终止和不明原因异常这个情况,如果没有自定义handler,

线程管理组就被默认为报告异常的handler。

ThreadGroup 这个类就是实现了UncaughtExceptionHandler这个接口,如果想捕获异常我们可以实现这个接口或者继承ThreadGroup,并重载uncaughtException方法。

在java API中对该接口描述的更详细:

我就不翻译了,太吃力了....%>_<%。在实现UncaughtExceptionHandler时,必须重载uncaughtException(Thread thread, Throwable ex) ,如果我们没有实现该接口

也就是没有显示捕捉异常,则ex为空,否则ex不为空,thread 则为出异常的线程。

接下来上代码,实现UncaughtExceptionHandler接口,显示处理线程异常终止的情况:

[java] view
plain
copy

  1. public class UnCeHandler implements UncaughtExceptionHandler {
  2. private Thread.UncaughtExceptionHandler mDefaultHandler;
  3. public static final String TAG = "CatchExcep";
  4. CatchExcep application;
  5. public UnCeHandler(CatchExcep application){
  6. //获取系统默认的UncaughtException处理器
  7. mDefaultHandler = Thread.getDefaultUncaughtExceptionHandler();
  8. this.application = application;
  9. }
  10. @Override
  11. public void uncaughtException(Thread thread, Throwable ex) {
  12. if(!handleException(ex) && mDefaultHandler != null){
  13. //如果用户没有处理则让系统默认的异常处理器来处理
  14. mDefaultHandler.uncaughtException(thread, ex);
  15. }else{
  16. try{
  17. Thread.sleep(2000);
  18. }catch (InterruptedException e){
  19. Log.e(TAG, "error : ", e);
  20. }
  21. Intent intent = new Intent(application.getApplicationContext(), MainActivity.class);
  22. PendingIntent restartIntent = PendingIntent.getActivity(
  23. application.getApplicationContext(), 0, intent,
  24. Intent.FLAG_ACTIVITY_NEW_TASK);
  25. //退出程序
  26. AlarmManager mgr = (AlarmManager)application.getSystemService(Context.ALARM_SERVICE);
  27. mgr.set(AlarmManager.RTC, System.currentTimeMillis() + 1000,
  28. restartIntent); // 1秒钟后重启应用
  29. application.finishActivity();
  30. }
  31. }
  32. /**
  33. * 自定义错误处理,收集错误信息 发送错误报告等操作均在此完成.
  34. *
  35. * @param ex
  36. * @return true:如果处理了该异常信息;否则返回false.
  37. */
  38. private boolean handleException(Throwable ex) {
  39. if (ex == null) {
  40. return false;
  41. }
  42. //使用Toast来显示异常信息
  43. new Thread(){
  44. @Override
  45. public void run() {
  46. Looper.prepare();
  47. Toast.makeText(application.getApplicationContext(), "很抱歉,程序出现异常,即将退出.",
  48. Toast.LENGTH_SHORT).show();
  49. Looper.loop();
  50. }
  51. }.start();
  52. return true;
  53. }
  54. }

通过在android Application 这个全局类中处理异常,如果不知道Application的作用请查看一下此链接:Application
详解

[java] view
plain
copy

  1. public class CatchExcep extends Application{
  2. ArrayList<Activity> list = new ArrayList<Activity>();
  3. public void init(){
  4. //设置该CrashHandler为程序的默认处理器
  5. UnCeHandler catchExcep = new UnCeHandler(this);
  6. Thread.setDefaultUncaughtExceptionHandler(catchExcep);
  7. }
  8. /**
  9. * Activity关闭时,删除Activity列表中的Activity对象*/
  10. public void removeActivity(Activity a){
  11. list.remove(a);
  12. }
  13. /**
  14. * 向Activity列表中添加Activity对象*/
  15. public void addActivity(Activity a){
  16. list.add(a);
  17. }
  18. /**
  19. * 关闭Activity列表中的所有Activity*/
  20. public void finishActivity(){
  21. for (Activity activity : list) {
  22. if (null != activity) {
  23. activity.finish();
  24. }
  25. }
  26. //杀死该应用进程
  27. android.os.Process.killProcess(android.os.Process.myPid());
  28. }
  29. }

然后人为制造一个异常:

[java] view
plain
copy

  1. Button btn;
  2. TextView tv;
  3. private CatchExcep application;
  4. @Override
  5. protected void onCreate(Bundle savedInstanceState) {
  6. super.onCreate(savedInstanceState);
  7. setContentView(R.layout.activity_main);
  8. btn = (Button)findViewById(R.id.btn);
  9. tv = (TextView)findViewById(R.id.tv);
  10. application = (CatchExcep)getApplication();
  11. application.init();
  12. application.addActivity(this);
  13. btn.setOnClickListener(this);
  14. }
  15. /**
  16. * 人为制造的异常*/
  17. public void press(){
  18. new Thread(new Runnable() {
  19. @Override
  20. public void run() {
  21. tv.setText("dfsd");
  22. }
  23. }).start();
  24. }
  25. @Override
  26. public void onClick(View v) {
  27. press();
  28. }
  29. }

上诉代码就能够实现 应用出现无法捕捉的异常时,杀死当前进程,重新启动一个应用。

我之前困扰的地方:搜了很多资料,杀死异常进程,重新启动应用,网上应用都是通过Application对象调用startActivity(intent),然后杀死异常进程。但是我怎样试都不成功,

进程是杀死了,但是应用却没启动起来,如果不将异常进程杀死,那么关闭应用时就得关闭两次,显然不能够接受。网上的一些方法都是错误的:如下几篇博客:

http://blog.csdn.net/xianming01/article/details/7711160

http://blog.csdn.net/ryantang03/article/details/9336295?reload

他们的方法能够捕获异常,杀死异常进程,但是却不能够重新启动应用。

如何杀死异常进程,重启应用,就得使用PendingIntent,这个类是android中对Intent类的包装,具体了解我会在写一篇博客,自己也可以去查看android API。

通过AlarmManager 启动它,并且关闭打开的Activity杀死异常进程就能够实现重新启动应用。

参考链接:

http://zheyiw.iteye.com/blog/1670990

android程序崩溃后重启的更多相关文章

  1. Android 程序崩溃后的处理

    在应用发布以后,由于安卓机型的千差万别 ,可能会出现各种各样的问题,这时候如果我们可以将这些信息收集起来,并进行修改就很不错了.下面就来讨论一下怎么处理程序崩溃以后,错误信息的手机. Java中已经提 ...

  2. 捕android程序崩溃日志

    主要类别: package com.example.callstatus; import java.io.File; import java.io.FileOutputStream; import j ...

  3. android 程序崩溃crash日志的捕捉

    android 程序崩溃crash日志的捕捉 之前在项目开发过程中,一直会遇到程序崩溃了,但是测试組的哥哥们又没及时的导出日志.... 后来在诳群的时候听别人说起,腾讯有那么一个叫bugly的东西 将 ...

  4. 结合程序崩溃后的core文件分析bug

    引言     在<I/O的效率比较>中,我们在修改图1程序的BUF_SIZE为8388608时,运行程序出现崩溃,如下图1:          图1. 段错误     一般而言,导致程序段 ...

  5. Android程序崩溃异常收集框架

    最近在写Android程序崩溃异常处理,完成之后,稍加封装与大家分享. 我的思路是这样的,在程序崩溃之后,将异常信息保存到一个日志文件中,然后对该文件进行处理,比如发送到邮箱,或发送到服务器. 所以, ...

  6. android双进程守护,让程序崩溃后一定可以重启

    由于我们做的是机器人上的软件,而机器人是24小时不间断服务的,这就要求我们的软件不能退出到系统桌面.当然最好是能够做到程序能够不卡顿,不崩溃,自己不退出.由于我们引用了很多第三方的开发包,也不能保证他 ...

  7. Android 捕获异常并在应用崩溃后重启应用

    问题概述: 在Android应用开发中,偶尔会因为测试的不充分导致一些异常没有被捕获,这时应用会出现异常并强制关闭,这样会导致很不好的用户体验,为了解决这个问题,我们需要捕获相关的异常并做处理. 首先 ...

  8. Android应用崩溃后异常捕获并重启并写入日志

    在Android开发时,有时会因为一些异常导致应用报错,偶尔会因为错误 而崩溃,导致用户体验下降,为了解决这问题,我们就要对这样的异常处理: 代码如下: CrashHandler.java impor ...

  9. linux如何让一个程序崩溃后自动重启

    思路:  写一个脚本 监控程序的运行状态  没有运行启动运行 已运行不做操作. 如果在控制台启动脚本 注意必须  nohup sh xxx.sh & while true do ps -ef ...

随机推荐

  1. hdu 3948 后缀数组

    The Number of Palindromes Time Limit: 6000/3000 MS (Java/Others)    Memory Limit: 262144/262144 K (J ...

  2. 【bzoj4570 scoi2016】妖怪

    题目描述 邱老师是妖怪爱好者,他有n只妖怪,每只妖怪有攻击力atk和防御力dnf两种属性.邱老师立志成为妖怪大师,于是他从真新镇出发,踏上未知的旅途,见识不同的风景. 环境对妖怪的战斗力有很大影响,在 ...

  3. zoj2112 树状数组+主席树 区间动第k大

    Dynamic Rankings Time Limit: 10000MS   Memory Limit: 32768KB   64bit IO Format: %lld & %llu Subm ...

  4. [BZOJ]4650 优秀的拆分(Noi2016)

    比较有意思的一道后缀数组题.(小C最近是和后缀数组淦上了?) 放在NOI的考场上.O(n^3)暴力80分,O(n^2)暴力95分…… 即使想把它作为一道签到题也不要这么随便啊摔(╯‵□′)╯︵┻━┻ ...

  5. A Problem-Solving FlowChart || 如何解决编程问题

    This is from book Cracking the coding interview, Gayle Laakmann Mcdowell. The flowchart can be used ...

  6. 《Java技术》第一次作业——Java语言基础

    学习总结 Scanner类实现基本数据输入的方法 Scanner 使用分隔符模式将其输入分解为标记,默认情况下该分隔符模式与空白匹配.然后可以使用不同的 next 方法将得到的标记转换为不同类型的值. ...

  7. 备忘:MySQL中修改表中某列的数据类型、删除外键约束

    -- MySQL中修改表中某列的数据类型 ALTER TABLE [COLUMN] 表名 MODIFY 列名 列定义; -- 删除外键约束 SHOW CREATE TABLE 表名; -- 复制CON ...

  8. Spring常用接口和类

    一.ApplicationContextAware接口 当一个类需要获取ApplicationContext实例时,可以让该类实现ApplicationContextAware接口.代码展示如下: p ...

  9. 设计模式:HelloWorld之策略模式

    一.概述 策略模式 定义了算法族,分别封装起来,让他们可以互相替换,此模式让算法的变化独立于使用算法的客户. 策略模式的三要素: 抽象策略角色: 策略类,通常由一个接口或者抽象类实现. 具体策略角色: ...

  10. 未能加载 global.asax的类的解决方案

    “/suitecallback”应用程序中的服务器错误. 分析器错误 说明: 在分析向此请求提供服务所需资源时出错.请检查下列特定分析错误详细信息并适当地修改源文件. 分析器错误消息: 未能加载类型“ ...