有时候由于测试不充分或者程序潜在的问题而导致程序异常崩溃,这个是令人无法接受的,在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. poj 2886 线段树+反素数

    Who Gets the Most Candies? Time Limit: 5000MS   Memory Limit: 131072K Total Submissions: 12744   Acc ...

  2. WebStorm配置node.js调试

    最近因为工作关系,一直在做node.js的开发,学习了koa框架,orm框架sequelize,以及swagger文档的配置.但是,最近因为swagger文档使用了es6的修饰器那么个东西(在java ...

  3. Linux必备操作vim

    vim被称作为编辑器之神,那么在我们操作linux系统时,进行编辑操作有没有感觉心有余而力不足?今天我讲自己总结的一些vim的操作命令和大家进行一下分享,有不足之处还请指出. vim的三种模式大家还记 ...

  4. Windows 2008 R2_NLB网络负载均衡(图文详解)(转)

    目录 前言 软件环境 DNS域名服务器 DNS服务器原理 DNS域名空间 DNS区域 DNS服务器的类别 DNS查询模式 缓存文件 配置DNS服务器 DNS服务的应用 新建子域 在DNS正向解析中新建 ...

  5. Python开发——排队问题随机模拟分析

    案例:主要是基于"蒙特卡罗思想",求解排队等待时间问题 场景:厕所排队问题 1.两场电影结束时间相隔较长,互不影响: 2.每场电影结束之后会有20个人想上厕所: 3.这20个人会在 ...

  6. Java的五子棋实现

    Java 五子棋 注:除机器人算法外其余借鉴于MLDN. package MyFiveChess; import robot.*; import java.awt.*; import javax.sw ...

  7. IScroll.js 学习笔记

    一.css部分1.transform 旋转div { transform:rotate(7deg); -ms-transform:rotate(7deg); /* IE 9 */ -moz-trans ...

  8. Java HashMap的扩容

    最近博主参加面试,发现自己对于Java的HashMap的扩容过程理解不足,故最近在此进行总结. 首先说明博主德Java为1.8版本 HashMap中的变量 首先要了解HashMap的扩容过程,我们就得 ...

  9. springboot解决跨域问题(Cors)

    1.对于前后端分离的项目来说,如果前端项目与后端项目部署在两个不同的域下,那么势必会引起跨域问题的出现. 针对跨域问题,我们可能第一个想到的解决方案就是jsonp,并且以前处理跨域问题我基本也是这么处 ...

  10. 日常实用css布局技巧汇总

    1.单行完整显示,多行省略显示. .box { width: 100px;  //必要 display: -webkit-box;    //必要 font-size: 14px; line-heig ...