德克云技术联盟

会员中心
发新帖
打印 上一主题 下一主题

Android中全局异常捕获
发布人: 李昭 发布时间:2014-11-04 浏览:3675

首先在全局变量中的onCreate事件中增加捕获入口:

MyPara.java

package deco.sfe.global;

import java.util.HashMap;

import android.app.Application;

public class MyPara extends Application{
   
    /**全局变量*/
    private String userid;
    private String user;
    private String pwd;
    private String orgid;
    private String token;
    private String ver;
    private String release_date;
    private String conpany;
    private String dfam_path;
    private String service_time;
    private String MIME;
    private String uploadUrl;//拍照图片上传地址
    private boolean isRememberPwd=false;
    private HashMap<String,String> services=new HashMap<String, String>();
   
    public void addService(String serviceName,String serviceImplName){
        services.put(serviceName, serviceImplName);
    }
   
    public String getService(String serviceName){
        return services.get(serviceName);
    }
   
    public String getUser() {
        return user;
    }
    public void setUser(String user) {
        this.user = user;
    }
    public String getPwd() {
        return pwd;
    }
    public void setPwd(String pwd) {
        this.pwd = pwd;
    }
    public String getOrgid() {
        return orgid;
    }
    public void setOrgid(String orgid) {
        this.orgid = orgid;
    }
    public String getVer() {
        return ver;
    }
    public void setVer(String ver) {
        this.ver = ver;
    }
    public String getRelease_date() {
        return release_date;
    }
    public void setRelease_date(String release_date) {
        this.release_date = release_date;
    }
    public String getConpany() {
        return conpany;
    }
    public void setConpany(String conpany) {
        this.conpany = conpany;
    }
    public String getDfam_path() {
        return dfam_path;
    }
    public void setDfam_path(String dfam_path) {
        this.dfam_path = dfam_path;
    }
    public String getService_time() {
        return service_time;
    }
    public void setService_time(String service_time) {
        this.service_time = service_time;
    }

    public String getMIME() {
        return MIME;
    }

    public void setMIME(String mIME) {
        MIME = mIME;
    }

    public boolean isRememberPwd() {
        return isRememberPwd;
    }

    public void setRememberPwd(boolean isRememberPwd) {
        this.isRememberPwd = isRememberPwd;
    }

    public String getToken() {
        return token;
    }

    public void setToken(String token) {
        this.token = token;
    }

    public String getUserid() {
        return userid;
    }

    public void setUserid(String userid) {
        this.userid = userid;
    }

    public String getUploadUrl() {
        return uploadUrl;
    }

    public void setUploadUrl(String uploadUrl) {
        this.uploadUrl = uploadUrl;
    }
   
    @Override  
    public void onCreate() {  
        super.onCreate();  
        ExceptionHandler exceptionHandler = ExceptionHandler.getInstance();  
        exceptionHandler.init(getApplicationContext());  
    }
}


然后定义句柄:ExceptionHandler.java
package deco.sfe.global;

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.widget.Toast;

public class ExceptionHandler implements UncaughtExceptionHandler {
   
    //系统默认的UncaughtException处理类   
    private Thread.UncaughtExceptionHandler mDefaultHandler;
   
    //ExceptionHandler实例  
    private static ExceptionHandler INSTANCE = new ExceptionHandler();  
   
    //程序的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 ExceptionHandler() {  
    }  
   
    /** 获取CrashHandler实例 ,单例模式 */  
    public static ExceptionHandler 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());  
                SysApplication.getInstance().exit();
            }  
    }
   
    /**
     * 自定义错误处理,收集错误信息 发送错误报告等操作均在此完成.
     *  
     * @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;  
    }  
}


完了。
其中封闭了“收集设备参数信息 ”、“写日志”的方法。具体应用中可以放开或者完善、优化。  



分类浏览
关于我们
联系我们
技术联盟
云服务
云技术
云合作
帮助中心
发帖规则
QQ客服
内部通道
企业邮箱
企业论坛
版本选择
手机版
电脑版
用手机扫描下方二维码查看手机版页面

版权所有 西安云联电子科技有限公司
Copyright @ 2011-2022 | decoclouds.com All Rig
陕ICP备13002202号-1

快速回复 返回顶部 返回列表