android 捕获未处理的异常

是从别处转来的,不过时间长了,忘了原文出处。


public class UncaughtExceptionHandler implements
		Thread.UncaughtExceptionHandler {
	
		/** Debug Log tag*/    
	    public static final String TAG = "jbl_exception";    
	    /** 是否开启日志输出,在Debug状态下开启,  
	     * 在Release状态下关闭以提示程序性能  
	     * */    
	    public static final boolean DEBUG = true;    
	    /** 系统默认的UncaughtException处理类 */    
	    private Thread.UncaughtExceptionHandler mDefaultHandler;    
	    /** CrashHandler实例 */    
	    private static UncaughtExceptionHandler INSTANCE;    
	    /** 程序的Context对象 */    
	    private Context mContext;    
	        
	    /** 使用Properties来保存设备的信息和错误堆栈信息*/    
	    private Properties mDeviceCrashInfo = new Properties();    
	    private static final String VERSION_NAME = "versionName";    
	    private static final String VERSION_CODE = "versionCode";    
	    private static final String STACK_TRACE = "STACK_TRACE";    
	    /** 错误报告文件的扩展名 */    
	    private static final String CRASH_REPORTER_EXTENSION = ".cr";    
	        
	    /** 保证只有一个CrashHandler实例 */    
	    private UncaughtExceptionHandler() {}    
	    /** 获取CrashHandler实例 ,单例模式*/    
	    public static UncaughtExceptionHandler getInstance() {    
	        if (INSTANCE == null) {    
	            INSTANCE = new UncaughtExceptionHandler();    
	        }    
	        return INSTANCE;
	    }    
	    
	    /**  
	     * 初始化,注册Context对象,  
	     * 获取系统默认的UncaughtException处理器,  
	     * 设置该CrashHandler为程序的默认处理器  
	     *   
	     * @param ctx  
	     */    
	    public void init(Context ctx) {    
	        mContext = ctx;    
	        mDefaultHandler = Thread.getDefaultUncaughtExceptionHandler();    
	        Thread.setDefaultUncaughtExceptionHandler(this);    
	    }    
	    
	    /**  
	     * 当UncaughtException发生时会转入该函数来处理  
	     */    
 
	    public void uncaughtException(Thread thread, Throwable ex) {    
	        if (!handleException(ex) && mDefaultHandler != null) {    
	            //如果用户没有处理则让系统默认的异常处理器来处理    
	            mDefaultHandler.uncaughtException(thread, ex);    
	        } else {    
	            //Sleep一会后结束程序    
	            try {    
	                Thread.sleep(3000);    
	            } catch (InterruptedException e) {    
	                Log.e(TAG, "Error : ", e);    
	            }    
	            android.os.Process.killProcess(android.os.Process.myPid());    
	            System.exit(10);    
	        }    
	    }    
	    
	    /**  
	     * 自定义错误处理,收集错误信息  
	     * 发送错误报告等操作均在此完成.  
	     * 开发者可以根据自己的情况来自定义异常处理逻辑  
	     * @param ex  
	     * @return true:如果处理了该异常信息;否则返回false  
	     */    
	    private boolean handleException(Throwable ex) {    
	        if (ex == null) {    
	            return true;    
	        }    
	        final String msg = ex.getLocalizedMessage();    
	        //使用Toast来显示异常信息    
	        new Thread() {    
	            @Override    
	            public void run() {    
	                Looper.prepare();    
	                Toast.makeText(mContext, "程序出错啦:" + msg, Toast.LENGTH_LONG)    
	                        .show();    
	                Looper.loop();    
	            }    
	    
	        }.start();    
	        //收集设备信息    
	        collectCrashDeviceInfo(mContext);    
	        //保存错误报告文件    
	        String crashFileName = saveCrashInfoToFile(ex);    
	        //发送错误报告到服务器    
//	        sendCrashReportsToServer(mContext);    
	        return true;    
	    }    
	    
	    /**  
	     * 在程序启动时候, 可以调用该函数来发送以前没有发送的报告  
	     */    
	    public void sendPreviousReportsToServer() {    
	        sendCrashReportsToServer(mContext);    
	    }    
	    
	    /**  
	     * 把错误报告发送给服务器,包含新产生的和以前没发送的.  
	     *   
	     * @param ctx  
	     */    
	    private void sendCrashReportsToServer(Context ctx) {    
	        String[] crFiles = getCrashReportFiles(ctx);    
	        if (crFiles != null && crFiles.length > 0) {    
	            TreeSet<String> sortedFiles = new TreeSet<String>();    
	            sortedFiles.addAll(Arrays.asList(crFiles));    
	    
	            for (String fileName : sortedFiles) {    
	                File cr = new File(ctx.getFilesDir(), fileName);    
	                postReport(cr);    
	                cr.delete();// 删除已发送的报告    
	            }    
	        }    
	    }    
	    
	    private void postReport(File file) {    
	        // TODO 使用HTTP Post 发送错误报告到服务器    
	        // 这里不再详述,开发者可以根据OPhoneSDN上的其他网络操作    
	        // 教程来提交错误报告    
	    }    
	    
	    /**  
	     * 获取错误报告文件名  
	     * @param ctx  
	     * @return  
	     */    
	    private String[] getCrashReportFiles(Context ctx) {    
	        File filesDir = ctx.getFilesDir();    
	        FilenameFilter filter = new FilenameFilter() {    
	            public boolean accept(File dir, String name) {    
	                return name.endsWith(CRASH_REPORTER_EXTENSION);    
	            }    
	        };    
	        return filesDir.list(filter);    
	    }    
	    /**  
	     * 保存错误信息到文件中  
	     * @param ex  
	     * @return  
	     */    
	    private String saveCrashInfoToFile(Throwable ex) {    
	        Writer info = new StringWriter();    
	        PrintWriter printWriter = new PrintWriter(info);    
	        ex.printStackTrace(printWriter);    
	    
	        Throwable cause = ex.getCause();    
	        while (cause != null) {    
	            cause.printStackTrace(printWriter);    
	            cause = cause.getCause();    
	        }    
	    
	        String result = info.toString();    
	        printWriter.close();    
	        mDeviceCrashInfo.put(STACK_TRACE, result);    
	    
	        try {    
	            long timestamp = System.currentTimeMillis();    
	            String fileName = "crash-" + timestamp + CRASH_REPORTER_EXTENSION;    
	            FileOutputStream trace = mContext.openFileOutput(fileName,    
	                    Context.MODE_PRIVATE);    
	            mDeviceCrashInfo.store(trace, "");    
	            trace.flush();    
	            trace.close();    
	            return fileName;    
	        } catch (Exception e) {    
	            Log.e(TAG, "an error occured while writing report file...", e);    
	        }    
	        return null;    
	    }    
	    
	    
	    /**  
	     * 收集程序崩溃的设备信息  
	     *   
	     * @param ctx  
	     */    
	    public void collectCrashDeviceInfo(Context ctx) {    
	        try {    
	            PackageManager pm = ctx.getPackageManager();    
	            PackageInfo pi = pm.getPackageInfo(ctx.getPackageName(),    
	                    PackageManager.GET_ACTIVITIES);    
	            if (pi != null) {    
	                mDeviceCrashInfo.put(VERSION_NAME,    
	                        pi.versionName == null ? "not set" : pi.versionName);    
	                mDeviceCrashInfo.put(VERSION_CODE, pi.versionCode);    
	            }    
	        } catch (NameNotFoundException e) {    
	            Log.e(TAG, "Error while collect package info", e);    
	        }    
	        //使用反射来收集设备信息.在Build类中包含各种设备信息,    
	        //例如: 系统版本号,设备生产商 等帮助调试程序的有用信息    
	//具体信息请参考后面的截图    
	        Field[] fields = Build.class.getDeclaredFields();    
	        for (Field field : fields) {    
	            try {    
	                field.setAccessible(true);    
	                mDeviceCrashInfo.put(field.getName(), field.get(null));    
	                if (DEBUG) {    
	                    Log.d(TAG, field.getName() + " : " + field.get(null));    
	                }    
	            } catch (Exception e) {    
	                Log.e(TAG, "Error while collect crash info", e);    
	            }    
	    
	        }    
	    
	    }    
	
}



  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值