android基础知识24:Android中处理崩溃异常


 大家都知道,现在安装Android系统的手机版本和设备千差万别,在模拟器上运行良好的程序安装到某款手机上说不定就出现崩溃的现象,开发者个人不可能购买所有设备逐个调试,所以在程序发布出去之后,如果出现了崩溃现象,开发者应该及时获取在该设备上导致崩溃的信息,这对于下一个版本的bug修复帮助极大,所以今天就来介绍一下如何在程序崩溃的情况下收集相关的设备参数信息和具体的异常信息,并发送这些信息到服务器供开发者分析和调试程序。

1、重启activity

        通常情况下,如果Android应用出现未处理的异常,会出现下面类似的对话框,然后强制退出该应用:

 \
 


如果你想改变这种缺省的行为,比如出现未处理异常时显示自定义对话框,或是重启该应用,可以使用下面步骤重定义Android全局异常处理事件。
1. 实现Thread.UncaughtExceptionHandler 接口
一般可以通过派生Application类并实现Thread.UncaughtExceptionHandler 方法:

 

[java]
public class GNavigatorApplication extends Application   
    
implements   Thread.UncaughtExceptionHandler {   
    
...   
    
}   
public class GNavigatorApplication extends Application 
  
implements   Thread.UncaughtExceptionHandler { 
  
... 
  
}  2. 定义Thread.UncaughtExceptionHandler方法
比如重启某个Activity

 

[java]
public void uncaughtException(Thread thread, Throwable ex) {   
    
Intent intent = new Intent(this, GNavigatorActivity.class);   
    
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP    | Intent.FLAG_ACTIVITY_NEW_TASK);   
    
startActivity(intent);   
    
}   
public void uncaughtException(Thread thread, Throwable ex) { 
  
Intent intent = new Intent(this, GNavigatorActivity.class); 
  
intent.addFlags(Intent.FLAG_ACTIVITY_CLEAR_TOP    | Intent.FLAG_ACTIVITY_NEW_TASK); 
  
startActivity(intent); 
  
}  3. 重置缺省的Exception处理函数,比如可以在OnCreate方法重设Exception处理方法。

 

[java]
@Override   
    
public void onCreate() {   
    
...   
    
Thread.setDefaultUncaughtExceptionHandler(this);   
    
}   
@Override 
  
public void onCreate() { 
  
... 
  
Thread.setDefaultUncaughtExceptionHandler(this); 
  
}  这样在应用中出现未处理异常时,会自动重启应用,而不会出现Force Close对话框。
2、获取异常信息并上传至服务器

我们先建立一个crash项目,项目结构如图:

 \
 

 

在MainActivity.java代码中,代码是这样写的:

 

[java]
package com.scott.crash;   
   
import android.app.Activity;   
import android.os.Bundle;   
   
public class MainActivity extends Activity {   
   
    private String s;   
       
    @Override   
    public void onCreate(Bundle savedInstanceState) {   
        super.onCreate(savedInstanceState);   
        System.out.println(s.equals("any string"));   
    }   
}   
package com.scott.crash; 
 
import android.app.Activity; 
import android.os.Bundle; 
 
public class MainActivity extends Activity { 
 
    private String s; 
     
    @Override 
    public void onCreate(Bundle savedInstanceState) { 
        super.onCreate(savedInstanceState); 
        System.out.println(s.equals("any string")); 
    } 
}  我们在这里故意制造了一个潜在的运行期异常,当我们运行程序时就会出现以下界面:
\
 

遇到软件没有捕获的异常之后,系统会弹出这个默认的强制关闭对话框。
我们当然不希望用户看到这种现象,简直是对用户心灵上的打击,而且对我们的bug的修复也是毫无帮助的。我们需要的是软件有一个全局的异常捕获器,当出现一个我们没有发现的异常时,捕获这个异常,并且将异常信息记录下来,上传到服务器公开发这分析出现异常的具体原因。
接下来我们就来实现这一机制,不过首先我们还是来了解以下两个类:android.app.Application和java.lang.Thread.UncaughtExceptionHandler。
Application:用来管理应用程序的全局状态。在应用程序启动时Application会首先创建,然后才会根据情况(Intent)来启动相应的Activity和Service。本示例中将在自定义加强版的Application中注册未捕获异常处理器。
Thread.UncaughtExceptionHandler:线程未捕获异常处理器,用来处理未捕获异常。如果程序出现了未捕获异常,默认会弹出系统中强制关闭对话框。我们需要实现此接口,并注册为程序中默认未捕获异常处理。这样当未捕获异常发生时,就可以做一些个性化的异常处理操作。
大家刚才在项目的结构图中看到的CrashHandler.java实现了Thread.UncaughtExceptionHandler,使我们用来处理未捕获异常的主要成员,代码如下:

 

[java]
package com.scott.crash;   
   
import java.io.File;   
import java.io.FileOutputStream;   
import java.io.PrintWriter;   
import java.io.StringWriter;   
import java.io.Writer;   
import java.lang.Thread.UncaughtExceptionHandler;   
import java.lang.reflect.Field;   
import java.text.DateFormat;   
import java.text.SimpleDateFormat;   
import java.util.Date;   
import java.util.HashMap;   
import java.util.Map;   
   
import android.content.Context;   
import android.content.pm.PackageInfo;   
import android.content.pm.PackageManager;   
import android.content.pm.PackageManager.NameNotFoundException;   
import android.os.Build;   
import android.os.Environment;   
import android.os.Looper;   
import android.util.Log;   
import android.widget.Toast;   
   
/** 
 * UncaughtException处理类,当程序发生Uncaught异常的时候,有该类来接管程序,并记录发送错误报告. 
 *  
 * @author user 
 *  
 */   
public class CrashHandler implements UncaughtExceptionHandler {   
       
    public static final String TAG = "CrashHandler";   
       
    //系统默认的UncaughtException处理类     
    private Thread.UncaughtExceptionHandler mDefaultHandler;   
    //CrashHandler实例    
    private static CrashHandler INSTANCE = new CrashHandler();   
    //程序的Context对象    
    private Context mContext;   
    //用来存储设备信息和异常信息    
    private Map<String, String> infos = new HashMap<String, String>();   
   
    //用于格式化日期,作为日志文件名的一部分    
    private DateFormat formatter = new SimpleDateFormat("yyyy-MM-dd-HH-mm-ss");   
   
    /** 保证只有一个CrashHandler实例 */   
    private CrashHandler() {   
    }   
   
    /** 获取CrashHandler实例 ,单例模式 */   
    public static CrashHandler getInstance() {   
        return INSTANCE;   
    }   
   
    /** 
     * 初始化 
     *  
     * @param context 
     */   
    public void init(Context context) {   
        mContext = context;   
        //获取系统默认的UncaughtException处理器    
        mDefaultHandler = Thread.getDefaultUncaughtExceptionHandler();   
        //设置该CrashHandler为程序的默认处理器    
        Thread.setDefaultUncaughtExceptionHandler(this);   
    }   
   
    /** 
     * 当UncaughtException发生时会转入该函数来处理 
     */   
    @Override   
    public void uncaughtException(Thread thread, Throwable ex) {   
        if (!handleException(ex) && mDefaultHandler != null) {   
            //如果用户没有处理则让系统默认的异常处理器来处理    
            mDefaultHandler.uncaughtException(thread, ex);   
        } else {   
            try {   
                Thread.sleep(3000);   
            } catch (InterruptedException e) {   
                Log.e(TAG, "error : ", e);   
            }   
            //退出程序    
            android.os.Process.killProcess(android.os.Process.myPid());   
            System.exit(1);   
        }   
    }   
   
    /** 
     * 自定义错误处理,收集错误信息 发送错误报告等操作均在此完成. 
     *  
     * @param ex 
     * @return true:如果处理了该异常信息;否则返回false. 
     */   
    private boolean handleException(Throwable ex) {   
        if (ex == null) {   
            return false;   
        }   
        //使用Toast来显示异常信息    
        new Thread() {   
            @Override   
            public void run() {   
                Looper.prepare();   
                Toast.makeText(mContext, "很抱歉,程序出现异常,即将退出.", Toast.LENGTH_LONG).show();   
                Looper.loop();   
            }   
        }.start();   
        //收集设备参数信息     
        collectDeviceInfo(mContext);   
        //保存日志文件     
        saveCrashInfo2File(ex);   
        return true;   
    }   
       
    /** 
     * 收集设备参数信息 
     * @param ctx 
     */   
    public void collectDeviceInfo(Context ctx) {   
        try {   
            PackageManager pm = ctx.getPackageManager();   
            PackageInfo pi = pm.getPackageInfo(ctx.getPackageName(), PackageManager.GET_ACTIVITIES);   
            if (pi != null) {   
                String versionName = pi.versionName == null ? "null" : pi.versionName;   
                String versionCode = pi.versionCode + "";   
                infos.put("versionName", versionName);   
                infos.put("versionCode", versionCode);   
            }   
        } catch (NameNotFoundException e) {   
            Log.e(TAG, "an error occured when collect package info", e);   
        }   
        Field[] fields = Build.class.getDeclaredFields();   
        for (Field field : fields) {   
            try {   
                field.setAccessible(true);   
                infos.put(field.getName(), field.get(null).toString());   
                Log.d(TAG, field.getName() + " : " + field.get(null));   
            } catch (Exception e) {   
                Log.e(TAG, "an error occured when collect crash info", e);   
            }   
        }   
    }   
   
    /** 
     * 保存错误信息到文件中 
     *  
     * @param ex 
     * @return  返回文件名称,便于将文件传送到服务器 
     */   
    private String saveCrashInfo2File(Throwable ex) {   
           
        StringBuffer sb = new StringBuffer();   
        for (Map.Entry<String, String> entry : infos.entrySet()) {   
            String key = entry.getKey();   
            String value = entry.getValue();   
            sb.append(key + "=" + value + "\n");   
        }   
           
        Writer writer = new StringWriter();   
        PrintWriter printWriter = new PrintWriter(writer);   
        ex.printStackTrace(printWriter);   
        Throwable cause = ex.getCause();   
        while (cause != null) {   
            cause.printStackTrace(printWriter);   
            cause = cause.getCause();   
        }   
        printWriter.close();   
        String result = writer.toString();   
        sb.append(result);   
        try {   
            long timestamp = System.currentTimeMillis();   
            String time = formatter.format(new Date());   
            String fileName = "crash-" + time + "-" + timestamp + ".log";   
            if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) {   
                String path = "/sdcard/crash/";   
                File dir = new File(path);   
                if (!dir.exists()) {   
                    dir.mkdirs();   
                }   
                FileOutputStream fos = new FileOutputStream(path + fileName);   
                fos.write(sb.toString().getBytes());   
                fos.close();   
            }   
            return fileName;   
        } catch (Exception e) {   
            Log.e(TAG, "an error occured while writing file...", e);   
        }   
        return null;   
    }   
}   
package com.scott.crash; 
 
import java.io.File; 
import java.io.FileOutputStream; 
import java.io.PrintWriter; 
import java.io.StringWriter; 
import java.io.Writer; 
import java.lang.Thread.UncaughtExceptionHandler; 
import java.lang.reflect.Field; 
import java.text.DateFormat; 
import java.text.SimpleDateFormat; 
import java.util.Date; 
import java.util.HashMap; 
import java.util.Map; 
 
import android.content.Context; 
import android.content.pm.PackageInfo; 
import android.content.pm.PackageManager; 
import android.content.pm.PackageManager.NameNotFoundException; 
import android.os.Build; 
import android.os.Environment; 
import android.os.Looper; 
import android.util.Log; 
import android.widget.Toast; 
 
/**
 * UncaughtException处理类,当程序发生Uncaught异常的时候,有该类来接管程序,并记录发送错误报告.
 * 
 * @author user
 * 
 */ 
public class CrashHandler implements UncaughtExceptionHandler { 
     
    public static final String TAG = "CrashHandler"; 
     
    //系统默认的UncaughtException处理类  
    private Thread.UncaughtExceptionHandler mDefaultHandler; 
    //CrashHandler实例 
    private static CrashHandler INSTANCE = new CrashHandler(); 
    //程序的Context对象 
    private Context mContext; 
    //用来存储设备信息和异常信息 
    private Map<String, String> infos = new HashMap<String, String>(); 
 
    //用于格式化日期,作为日志文件名的一部分 
    private DateFormat formatter = new SimpleDateFormat("yyyy-MM-dd-HH-mm-ss"); 
 
    /** 保证只有一个CrashHandler实例 */ 
    private CrashHandler() { 
    } 
 
    /** 获取CrashHandler实例 ,单例模式 */ 
    public static CrashHandler getInstance() { 
        return INSTANCE; 
    } 
 
    /**
     * 初始化
     * 
     * @param context
     */ 
    public void init(Context context) { 
        mContext = context; 
        //获取系统默认的UncaughtException处理器 
        mDefaultHandler = Thread.getDefaultUncaughtExceptionHandler(); 
        //设置该CrashHandler为程序的默认处理器 
        Thread.setDefaultUncaughtExceptionHandler(this); 
    } 
 
    /**
     * 当UncaughtException发生时会转入该函数来处理
     */ 
    @Override 
    public void uncaughtException(Thread thread, Throwable ex) { 
        if (!handleException(ex) && mDefaultHandler != null) { 
            //如果用户没有处理则让系统默认的异常处理器来处理 
            mDefaultHandler.uncaughtException(thread, ex); 
        } else { 
            try { 
                Thread.sleep(3000); 
            } catch (InterruptedException e) { 
                Log.e(TAG, "error : ", e); 
            } 
            //退出程序 
            android.os.Process.killProcess(android.os.Process.myPid()); 
            System.exit(1); 
        } 
    } 
 
    /**
     * 自定义错误处理,收集错误信息 发送错误报告等操作均在此完成.
     * 
     * @param ex
     * @return true:如果处理了该异常信息;否则返回false.
     */ 
    private boolean handleException(Throwable ex) { 
        if (ex == null) { 
            return false; 
        } 
        //使用Toast来显示异常信息 
        new Thread() { 
            @Override 
            public void run() { 
                Looper.prepare(); 
                Toast.makeText(mContext, "很抱歉,程序出现异常,即将退出.", Toast.LENGTH_LONG).show(); 
                Looper.loop(); 
            } 
        }.start(); 
        //收集设备参数信息  
        collectDeviceInfo(mContext); 
        //保存日志文件  
        saveCrashInfo2File(ex); 
        return true; 
    } 
     
    /**
     * 收集设备参数信息
     * @param ctx
     */ 
    public void collectDeviceInfo(Context ctx) { 
        try { 
            PackageManager pm = ctx.getPackageManager(); 
            PackageInfo pi = pm.getPackageInfo(ctx.getPackageName(), PackageManager.GET_ACTIVITIES); 
            if (pi != null) { 
                String versionName = pi.versionName == null ? "null" : pi.versionName; 
                String versionCode = pi.versionCode + ""; 
                infos.put("versionName", versionName); 
                infos.put("versionCode", versionCode); 
            } 
        } catch (NameNotFoundException e) { 
            Log.e(TAG, "an error occured when collect package info", e); 
        } 
        Field[] fields = Build.class.getDeclaredFields(); 
        for (Field field : fields) { 
            try { 
                field.setAccessible(true); 
                infos.put(field.getName(), field.get(null).toString()); 
                Log.d(TAG, field.getName() + " : " + field.get(null)); 
            } catch (Exception e) { 
                Log.e(TAG, "an error occured when collect crash info", e); 
            } 
        } 
    } 
 
    /**
     * 保存错误信息到文件中
     * 
     * @param ex
     * @return  返回文件名称,便于将文件传送到服务器
     */ 
    private String saveCrashInfo2File(Throwable ex) { 
         
        StringBuffer sb = new StringBuffer(); 
        for (Map.Entry<String, String> entry : infos.entrySet()) { 
            String key = entry.getKey(); 
            String value = entry.getValue(); 
            sb.append(key + "=" + value + "\n"); 
        } 
         
        Writer writer = new StringWriter(); 
        PrintWriter printWriter = new PrintWriter(writer); 
        ex.printStackTrace(printWriter); 
        Throwable cause = ex.getCause(); 
        while (cause != null) { 
            cause.printStackTrace(printWriter); 
            cause = cause.getCause(); 
        } 
        printWriter.close(); 
        String result = writer.toString(); 
        sb.append(result); 
        try { 
            long timestamp = System.currentTimeMillis(); 
            String time = formatter.format(new Date()); 
            String fileName = "crash-" + time + "-" + timestamp + ".log"; 
            if (Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)) { 
                String path = "/sdcard/crash/"; 
                File dir = new File(path); 
                if (!dir.exists()) { 
                    dir.mkdirs(); 
                } 
                FileOutputStream fos = new FileOutputStream(path + fileName); 
                fos.write(sb.toString().getBytes()); 
                fos.close(); 
            } 
            return fileName; 
        } catch (Exception e) { 
            Log.e(TAG, "an error occured while writing file...", e); 
        } 
        return null; 
    } 
}  在收集异常信息时,朋友们也可以使用Properties,因为Properties有一个很便捷的方法properties.store(OutputStream out, String comments),用来将Properties实例中的键值对外输到输出流中,但是在使用的过程中发现生成的文件中异常信息打印在同一行,看起来极为费劲,所以换成Map来存放这些信息,然后生成文件时稍加了些操作。
完成这个CrashHandler后,我们需要在一个Application环境中让其运行,为此,我们继承android.app.Application,添加自己的代码,CrashApplication.java代码如下:


[java]
package com.scott.crash;   
   
import android.app.Application;   
   
public class CrashApplication extends Application {   
    @Override   
    public void onCreate() {   
        super.onCreate();   
        CrashHandler crashHandler = CrashHandler.getInstance();   
        crashHandler.init(getApplicationContext());   
    }   
}   
package com.scott.crash; 
 
import android.app.Application; 
 
public class CrashApplication extends Application { 
    @Override 
    public void onCreate() { 
        super.onCreate(); 
        CrashHandler crashHandler = CrashHandler.getInstance(); 
        crashHandler.init(getApplicationContext()); 
    } 
}  最后,为了让我们的CrashApplication取代android.app.Application的地位,在我们的代码中生效,我们需要修改AndroidManifest.xml:


[html]
<application android:name=".CrashApplication" ...>   
</application>   
<application android:name=".CrashApplication" ...> 
</application>  因为我们上面的CrashHandler中,遇到异常后要保存设备参数和具体异常信息到SDCARD,所以我们需要在AndroidManifest.xml中加入读写SDCARD权限:


[html]
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>   
<uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"/>  搞定了上边的步骤之后,我们来运行一下这个项目:
\
 

看以看到,并不会有强制关闭的对话框出现了,取而代之的是我们比较有好的提示信息。
然后看一下SDCARD生成的文件:

 

用文本编辑器打开日志文件,看一段日志信息:


[html]
CPU_ABI=armeabi   
CPU_ABI2=unknown   
ID=FRF91   
MANUFACTURER=unknown   
BRAND=generic   
TYPE=eng   
......   
Caused by: java.lang.NullPointerException   
    at com.scott.crash.MainActivity.onCreate(MainActivity.java:13)   
    at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1047)   
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2627)   
    ... 11 more   
CPU_ABI=armeabi 
CPU_ABI2=unknown 
ID=FRF91 
MANUFACTURER=unknown 
BRAND=generic 
TYPE=eng 
...... 
Caused by: java.lang.NullPointerException 
    at com.scott.crash.MainActivity.onCreate(MainActivity.java:13) 
    at android.app.Instrumentation.callActivityOnCreate(Instrumentation.java:1047) 
    at android.app.ActivityThread.performLaunchActivity(ActivityThread.java:2627) 
    ... 11 more  这些信息对于开发者来说帮助极大,所以我们需要将此日志文件上传到服务器,有关文件上传的技术,请参照Android中使用HTTP服务相关介绍。
不过在使用HTTP服务之前,需要确定网络畅通,我们可以使用下面的方式判断网络是否可用:


[java]
/** 
     * 网络是否可用 
     *  
     * @param context 
     * @return 
     */   
    public static boolean isNetworkAvailable(Context context) {   
        ConnectivityManager mgr = (ConnectivityManager) context.getSystemService(Context.CONNECTIVITY_SERVICE);   
        NetworkInfo[] info = mgr.getAllNetworkInfo();   
        if (info != null) {   
            for (int i = 0; i < info.length; i++) {   
                if (info[i].getState() == NetworkInfo.State.CONNECTED) {   
                    return true;   
                }   
            }   
        }   
        return false;   
    }   
作者:xianming01
 

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
Android是一种基于Linux操作系统和开放源代码的移动设备操作系统,由Google开发和维护,是目前全球最流行的移动操作系统之一。以下是Android基础知识: 1. Android系统架构:Android系统架构分为四个层次,分别是应用层、应用框架层、系统运行库层和Linux内核层。 2. Android应用:Android应用可以通过Java编程语言开发,使用Android SDK(软件开发工具包)提供的API进行开发,也可以使用其他编程语言如C++、Python等。 3. Android应用生命周期:Android应用生命周期指应用从启动到终止的整个过程,包括四种状态:运行状态、暂停状态、停止状态和销毁状态。 4. Android布局:Android布局指应用各个控件的排列方式,包括线性布局、相对布局、表格布局等多种布局方式。 5. Android控件:Android控件是应用的基本元素,包括文本框、按钮、复选框、列表等各种控件。 6. Android Intent:Intent是Android用于启动应用组件、传递数据和启动服务等的机制。 7. Android存储:Android提供了多种存储方式,包括SharedPreferences、文件存储、SQLite数据库等。 8. Android网络编程:Android可以使用HttpURLConnection、HttpClient等工具进行网络编程,也可以使用第三方库如Volley、OkHttp等。 9. Android多线程编程:Android可以使用AsyncTask、Handler等工具进行多线程编程。 10. Android调试:Android可以使用Logcat、DDMS等工具进行调试,也可以使用Android Studio提供的调试功能。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值