Android全局异常捕获机制_安卓用户名重复异常如何捕获-程序员宅基地

技术标签: q'  Android  

 

文章背景 
程序猿或是程序媛们在开发Android项目的时候,经常出现各种奇葩的Crash,有可能是服务端返回数据的原因所造成的、也可能是客户端自己的原因。我个人认为出现Bug并不是那么的重要,快速定位问题才是解决问题的开始、如果我们有一个能够帮助我们快速定位异常的机制,那么不仅在开发的效率上提高,而且在维护成本也会在一定程度上降低成本。 
那么目前主流的第三方Bug分析框架有腾讯的Bugly和友盟都能够很好的统计分析各种Crash等Bug,但是在我们做项目中,难免面对的用户是政府或银行等客户,这类客户有一些硬性的的要求,就是不允许嵌入第三方的框架,有可能是安全方面的担心,怕被监控或是盗取数据等。 
另一方面来讲,现在的第三方库提供商,总说自己的sdk特牛逼、然而一下载下来一看就有几十兆、甚至有几百兆、如果嵌入太多的第三方sdk会增大apk大小、若是互联网产品,包越大越难在用户手机上存活。

文章目标 
为我们的项目提供一个异常捕获跟踪处理机制,我认为应包含捕获异常、写入异常数据到SD卡中、定时上传异常数据给服务端、服务端统计分析异常、最终目标为解决异常从而提高代码的健壮性。

下面提供一个客户端这边的异常处理类,先上个演示效果图:

效果图

异常信息打印效果:

异常信息打印效果图

日志写入效果图:

日志写入效果图

异常处理类:

    package com.majunbao.tools;

import android.content.Context;
import android.content.pm.PackageInfo;
import android.content.pm.PackageManager;
import android.os.Build;
import android.os.Looper;
import android.text.format.Time;
import android.util.Log;
import android.view.Gravity;
import android.widget.Toast;

import java.io.File;
import java.io.FileOutputStream;
import java.io.FilenameFilter;
import java.io.PrintWriter;
import java.io.StringWriter;
import java.io.Writer;
import java.lang.reflect.Field;
import java.util.Arrays;
import java.util.Properties;
import java.util.TreeSet;

/**
 * Created by Administrator on 2017/6/15.
 */

public class CrashHandler implements Thread.UncaughtExceptionHandler {

    /** Debug Log tag*/
    public static final String TAG = "CrashHandler";
    /** 是否开启日志输出,在Debug状态下开启,
     * 在Release状态下关闭以提示程序性能
     * */
    public static final boolean DEBUG = true;
    /** 系统默认的UncaughtException处理类 */
    private Thread.UncaughtExceptionHandler mDefaultHandler;
    /** CrashHandler实例 */
    private static CrashHandler 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";

    private static  Object syncRoot = new Object();

    /** 保证只有一个CrashHandler实例 */
    private CrashHandler() {}

    /** 获取CrashHandler实例 ,单例模式*/
    public static CrashHandler getInstance() {
       /* if (INSTANCE == null) {
            INSTANCE = new CrashHandler();
        }
        return INSTANCE;*/
        // 防止多线程访问安全,这里使用了双重锁
        if (INSTANCE == null)
        {

            synchronized (syncRoot)
            {

                if (INSTANCE == null)
                {
                    INSTANCE =  new CrashHandler();
                }
            }
        }
        return INSTANCE;
    }

    /**
     * 初始化,注册Context对象,
     * 获取系统默认的UncaughtException处理器,
     * 设置该CrashHandler为程序的默认处理器
     * @param ctx
     */
    public void init(Context ctx) {
        mContext = ctx;
        mDefaultHandler = Thread.getDefaultUncaughtExceptionHandler();
        Thread.setDefaultUncaughtExceptionHandler(this);
    }

    /**
     * 当UncaughtException发生时会转入该函数来处理
     */
    @Override
    public void uncaughtException(Thread thread, Throwable ex) {
        if (!handleException(ex) && mDefaultHandler != null) {
            //如果用户没有处理则让系统默认的异常处理器来处理
            mDefaultHandler.uncaughtException(thread, ex);
        } else {
            //Sleep一会后结束程序
            try {
                Thread.sleep(5000);
            } 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) {
            Log.w(TAG, "handleException --- ex==null");
            return true;
        }
        final String msg = ex.getLocalizedMessage();
        if(msg == null) {
            return false;
        }
        //使用Toast来显示异常信息
        new Thread() {
            @Override
            public void run() {
                Looper.prepare();

                if(DEBUG){
                    Log.d(TAG, "异常信息->"+msg);
                    Toast toast = Toast.makeText(mContext, "程序出错,即将退出:\r\n" + msg,
                            Toast.LENGTH_LONG);
                    toast.setGravity(Gravity.CENTER, 0, 0);
                    toast.show();

                    //保存错误报告文件
                    LogToFile.w("my",msg);**//这句话可以先注释掉,这是我单独写的一个log写入类,下面已提供了该类**
                }
//              MsgPrompt.showMsg(mContext, "程序出错啦", msg+"\n点确认退出");
                Looper.loop();
            }
        }.start();
        //收集设备信息
        collectCrashDeviceInfo(mContext);
        //保存错误报告文件
        //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 发送错误报告到服务器
    }

    /**
     * 获取错误报告文件名
     * @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("EXEPTION", ex.getLocalizedMessage());
        mDeviceCrashInfo.put(STACK_TRACE, result);
        try {
            //long timestamp = System.currentTimeMillis();
            Time t = new Time("GMT+8");
            t.setToNow(); // 取得系统时间
            int date = t.year * 10000 + t.month * 100 + t.monthDay;
            int time = t.hour * 10000 + t.minute * 100 + t.second;
            String fileName = "crash-" + date + "-" + time + 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 (PackageManager.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);
            }
        }
    }


}

使用方法:

package com.majunbao.application;

import android.app.Application;

import com.majunbao.application.tools.CrashHandler;
import com.majunbao.application.tools.LogToFile;

/**
 * Created by Administrator on 2017/6/15.
 */

public class MyApplication extends Application {
    private final static float HEAP_UTILIZATION = 0.75f;
    private final static int MIN_HEAP_SIZE = 6* 1024* 1024 ;
    @Override
    public void onCreate() {
        super.onCreate();

        // 异常处理,不需要处理时注释掉这两句即可!
        CrashHandler crashHandler = CrashHandler.getInstance();
        // 注册crashHandler
        crashHandler.init(getApplicationContext());


    }
}

开始测试: 
package com.example.yangdechengapplication;

import android.support.v7.app.AppCompatActivity; 
import android.os.Bundle; 
import android.view.View; 
import android.widget.Button;

import com.example.yangdechengapplication.tools.LogToFile;

public class MainActivity extends AppCompatActivity {

Button btn_crashHandler;
@Override
protected void onCreate(Bundle savedInstanceState) {
    super.onCreate(savedInstanceState);
    setContentView(R.layout.activity_main);

    btn_crashHandler=(Button)findViewById(R.id.btn_crashHandler);
    btn_crashHandler.setOnClickListener(new View.OnClickListener() {
        @Override
        public void onClick(View v) {

            int a= Integer.parseInt("q");//当用户点击按钮的时候特意写了一个异常
            //LogToFile.w("my","我在测试文件读写66666666666啊");
        }
    });
}

}

LogToFile(异常日志写入类,已单独抽取出来了,也算职责分明吧) 
注:写入日志需要在配置文件中加上文件读写权限

 <!--往sdcard中写入数据的权限 -->
    <uses-permission android:name="android.permission.WRITE_EXTERNAL_STORAGE"></uses-permission>
    <!--在sdcard中创建/删除文件的权限 -->
    <uses-permission android:name="android.permission.MOUNT_UNMOUNT_FILESYSTEMS"></uses-permission>
  • 1
  • 2
  • 3
  • 4
package com.com.majunbao.application.tools;

import android.content.Context;
import android.os.Environment;
import android.util.Log;

import java.io.BufferedWriter;
import java.io.File;
import java.io.FileNotFoundException;
import java.io.FileOutputStream;
import java.io.IOException;
import java.io.OutputStreamWriter;
import java.text.SimpleDateFormat;
import java.util.Date;
import java.util.Locale;

/**
 * Created by Administrator on 2017/6/15.
 */

public class LogToFile {

    //日志是否需要读写开关
    public static final boolean DEBUG_FLAG = false;
    private static String TAG = "LogToFile";

    private static String logPath = null;//log日志存放路径

    private static SimpleDateFormat dateFormat = new SimpleDateFormat("yyyy-MM-dd_HH-mm-ss", Locale.US);//日期格式;

    private static Date date = new Date();//因为log日志是使用日期命名的,使用静态成员变量主要是为了在整个程序运行期间只存在一个.log文件中;

    /**
     * 初始化,须在使用之前设置,最好在Application创建时调用
     *
     * @param context
     */
    public static void init(Context context) {
        logPath = getFilePath(context) + "/Logs";//获得文件储存路径,在后面加"/Logs"建立子文件夹
    }

    /**
     * 获得文件存储路径
     *
     * @return
     */
    private static String getFilePath(Context context) {

        if (Environment.MEDIA_MOUNTED.equals(Environment.MEDIA_MOUNTED) || !Environment.isExternalStorageRemovable()) {//如果外部储存可用
            return context.getExternalFilesDir(null).getPath();//获得外部存储路径,默认路径为 /storage/emulated/0/Android/data/com.waka.workspace.logtofile/files/Logs/log_2016-03-14_16-15-09.log
        } else {
            return context.getFilesDir().getPath();//直接存在/data/data里,非root手机是看不到的
        }
    }

    private static final char VERBOSE = 'v';

    private static final char DEBUG = 'd';

    private static final char INFO = 'i';

    private static final char WARN = 'w';

    private static final char ERROR = 'e';

    public static void v(String tag, String msg) {
        if(DEBUG_FLAG){
            writeToFile(VERBOSE, tag, msg);
        }

    }

    public static void d(String tag, String msg) {
        if(DEBUG_FLAG){
            writeToFile(DEBUG, tag, msg);
        }
    }

    public static void i(String tag, String msg) {
        if(DEBUG_FLAG){
            writeToFile(INFO, tag, msg);
        }

    }

    public static void w(String tag, String msg) {
        if(DEBUG_FLAG){
            writeToFile(WARN, tag, msg);
        }
    }

    public static void e(String tag, String msg) {
        if(DEBUG_FLAG){
            writeToFile(ERROR, tag, msg);
        }

    }

    /**
     * 将log信息写入文件中
     *
     * @param type
     * @param tag
     * @param msg
     */
    private static void writeToFile(char type, String tag, String msg) {

        if (null == logPath) {
            Log.e(TAG, "logPath == null ,未初始化LogToFile");
            return;
        }

        String fileName = logPath + "/log_" + dateFormat.format(new Date()) + ".log";//log日志名,使用时间命名,保证不重复
        String log = dateFormat.format(date) + " " + type + " " + tag + " " + msg + "\n";//log日志内容,可以自行定制

        //如果父路径不存在
        File file = new File(logPath);
        if (!file.exists()) {
            file.mkdirs();//创建父路径
        }

        FileOutputStream fos = null;//FileOutputStream会自动调用底层的close()方法,不用关闭
        BufferedWriter bw = null;
        try {

            fos = new FileOutputStream(fileName, true);//这里的第二个参数代表追加还是覆盖,true为追加,flase为覆盖
            bw = new BufferedWriter(new OutputStreamWriter(fos));
            bw.write(log);

        } catch (FileNotFoundException e) {
            e.printStackTrace();
        } catch (IOException e) {
            e.printStackTrace();
        } finally {
            try {
                if (bw != null) {
                    bw.close();//关闭缓冲流
                }
            } catch (IOException e) {
                e.printStackTrace();
            }
        }

    }
}
版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/nideyida1/article/details/82721229

智能推荐

组态软件定义_上位机软件专业英语-程序员宅基地

文章浏览阅读1.9k次。组态软件,上位机软件的一种。组态软件<=上位机软件。又称组态监控软件系统软件。译自英文SCADA,即 Supervisory Control and Data Acquisition(数据采集与监视控制)。它是指一些数据采集与过程控制的专用软件。它们处在自动控制系统监控层一级的软件平台和开发环境,使用灵活的组态方式,为用户提供快速构建工业自动控制系统监控功能的、通用层次的软件工具。组态软件的应用领域很广,可以应用于电力系统、给水系统、石油、化工等领域的数据采集与监视控制以及过程控制等诸多领域。在电力系_上位机软件专业英语

Navicat实现将数据模型导出SQL文件,迅速生成表_navicat 模型生成sql-程序员宅基地

文章浏览阅读8.8k次,点赞3次,收藏8次。Navicat实现将数据模型导出SQL文件,迅速生成表这次我们不讲高大上的代码,我们来说一说开发中一个实用的小技能。直接进入正题1.首先打开Navicat工具2.点击右上角的模型..._navicat 模型生成sql

社交类APP原型模板分享——微信_微信原型图-程序员宅基地

文章浏览阅读3.6k次。微信是一款社交类的APP应用——聊天软件,支持多人群聊。交互效果主要有滚动内容界面、选择组件触发按钮状态变化、点击下拉展开列表、点击弹出面板等交互操作。本原型由国产原型工具-Mockplus制作完成。先简单看看动图:点击这里,可以立即在线预览:微信原型模板再送上UI Flow大图:原型中有消息列表页面、通讯录列表界面、发现列表界面、我的个人资料列表界面、朋友圈界面以..._微信原型图

面试的一些技巧_竞选大队长时到台上演讲怎样不紧张端庄大方举止得体-程序员宅基地

文章浏览阅读423次。应试者要想在面试答辩中获得成功,必须注意以下几个问题:(一)淡化面试的成败意识应试者对于面试的成败,首先在思想上应注意淡化,要有一种“不以物喜,不以己悲”的超然态度。如果在面试中有这样的心态,才会处变不惊。如果只想到成功,不想到失败,那么在面试中一遇到意外情况,就会惊慌失措,一败涂地。(二)保持自信应试者在面试前树立了自信,在面试中也要始终保持自信,只有保持了自信,才能够在面试中始终保持高度的注意力、缜密的思维力、敏锐的判断力、充沛的精力,夺取答辩的胜利。(三)保持愉悦的精神状态愉悦的精神状态,_竞选大队长时到台上演讲怎样不紧张端庄大方举止得体

SpringBoot增删改查的返回值类型_springboot mapper查詢返回數字-程序员宅基地

文章浏览阅读2.1k次。select(查询)方法需要集合或者数组去装数据,而update(修改),delete(删除),insert(添加)方法是不需要设置返回类型的,它们都是默认返回一个int。/** * 查询水果 * @return */ @RequestMapping("/selFruit") public List<Fruit> selFruit(){ List<Fruit> list = fruitService...._springboot mapper查詢返回數字

项目实战:C/C++游戏:2048[C语言版]-程序员宅基地

文章浏览阅读4.2k次,点赞14次,收藏65次。项目实战:C/C++游戏:2048[C语言版]目录项目实战:C/C++游戏:2048[C语言版]1.编译环境2.项目运行效果3.思路简介:1.游戏规则2.核心算法4.主要源码:1.编译环境Win10专业版x64 VS2015这是2017年9或10月份写的 一个练手的,和上一篇Flappy Bird 是一起的, 留以后一个永久的回忆..._c++游戏

随便推点

opencv-python(五):图像的算数运算_python如何计算图像-程序员宅基地

文章浏览阅读225次。0. 注意图像的本质就是矩阵,所以要以矩阵的眼光来理解1. 加减法1.1 加法 img1+img2:将两幅图像对应位置的像素相加(尺寸类型相同) img1+50:将img1全部像素+50,提高了整体亮度1.2 减法 img1-img2:两幅相同尺寸的图相减 img-50:将img1全部像素-50,降低了整体亮度将img1全部像素+50,提高了整体亮度右边两幅图是左边两幅图相减做差的结果,对于左边两幅图的差别不易观察,但是做差后就很明显1.2 图像混合图像混合的数学公式:g_python如何计算图像

MAC下go get github.com/sbinet/go-python报错pkg-config: exec: "pkg-config": executable file not found in-程序员宅基地

文章浏览阅读2.5k次。1. pkg-config: exec: "pkg-config": executable file not found in $PATH解决方案:brew install pkg-config2.Package python-2.7 was not found in the pkg-config search path.Perhaps you should add the d..._pkg-config: exec: "pkg-config": executable file not found in $path

Kivy-iOS: 带着Kivy框架去开发iOS应用-程序员宅基地

文章浏览阅读321次,点赞3次,收藏10次。Kivy-iOS: 带着Kivy框架去开发iOS应用[项目链接]简介在移动设备的市场中,Android 和 iOS 是两个主要的操作系统。开发者们通常需要为这两种平台分别开发应用以覆盖更多的用户群。现在有一个叫做 Kivy 的 Python 框架,可以让您在一种语言和一套代码基础上构建跨平台的应用程序,包括了 Android 和 iOS。本文将向您介绍如何利用 Kivy-iOS 工具链帮助您..._kivy怎么调用ios底层硬件

Mac下Git本地仓库连接Github-程序员宅基地

文章浏览阅读196次。参考:http://blog.163.com/xianfuying%40126/blog/static/21960005201181482518631/转载于:https://www.cnblogs.com/z-xx/p/7484416.html_本地仓库连接新的git仓库mac

Python 全栈系列106 -调用百度地图api计算两点间距离和行车时间_百度api查2点之间的推荐路线距离-程序员宅基地

文章浏览阅读1.9k次。说明百度地图的api还是蛮大方的,反正个人是肯定够用的。没想到还有天气接口,以后也可以玩一玩。本篇实现计算两个地址间的距离。内容1 地址转为经纬度使用这个接口进行地址转换坐标。第一次提交报APP 服务禁用,这时候就进入App的设置,我比较怕麻烦,一次全勾上了。随意输入一个地址address ='上海浦东国际机场卫星厅'---resp = req.get(url_template % (address , ak ))resp.text'showLocation&&s_百度api查2点之间的推荐路线距离

如何在word中插入latex公式和伪代码_word插入latex代码块-程序员宅基地

文章浏览阅读4.8w次,点赞23次,收藏102次。如何在word中插入latex公式和伪代码在word中可以通过软件Aurora把latex公式直接搬到word里面,超级好用,安装完软件之后,word的工具栏会自动出现该选项。至于伪代码,只要复制,点击工具栏的 Paste from Tex 就可以直接出现,方便极了!把伪代码搬到word里面可能出现的问题有以下几点:1.主动的算法序号不见了,推荐语句如下,其中星号可以被替代为相应序号\renewco_word插入latex代码块

推荐文章

热门文章

相关标签