文章背景
程序猿或是程序媛们在开发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>
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();
}
}
}
}
文章浏览阅读1.9k次。组态软件,上位机软件的一种。组态软件<=上位机软件。又称组态监控软件系统软件。译自英文SCADA,即 Supervisory Control and Data Acquisition(数据采集与监视控制)。它是指一些数据采集与过程控制的专用软件。它们处在自动控制系统监控层一级的软件平台和开发环境,使用灵活的组态方式,为用户提供快速构建工业自动控制系统监控功能的、通用层次的软件工具。组态软件的应用领域很广,可以应用于电力系统、给水系统、石油、化工等领域的数据采集与监视控制以及过程控制等诸多领域。在电力系_上位机软件专业英语
文章浏览阅读8.8k次,点赞3次,收藏8次。Navicat实现将数据模型导出SQL文件,迅速生成表这次我们不讲高大上的代码,我们来说一说开发中一个实用的小技能。直接进入正题1.首先打开Navicat工具2.点击右上角的模型..._navicat 模型生成sql
文章浏览阅读3.6k次。微信是一款社交类的APP应用——聊天软件,支持多人群聊。交互效果主要有滚动内容界面、选择组件触发按钮状态变化、点击下拉展开列表、点击弹出面板等交互操作。本原型由国产原型工具-Mockplus制作完成。先简单看看动图:点击这里,可以立即在线预览:微信原型模板再送上UI Flow大图:原型中有消息列表页面、通讯录列表界面、发现列表界面、我的个人资料列表界面、朋友圈界面以..._微信原型图
文章浏览阅读423次。应试者要想在面试答辩中获得成功,必须注意以下几个问题:(一)淡化面试的成败意识应试者对于面试的成败,首先在思想上应注意淡化,要有一种“不以物喜,不以己悲”的超然态度。如果在面试中有这样的心态,才会处变不惊。如果只想到成功,不想到失败,那么在面试中一遇到意外情况,就会惊慌失措,一败涂地。(二)保持自信应试者在面试前树立了自信,在面试中也要始终保持自信,只有保持了自信,才能够在面试中始终保持高度的注意力、缜密的思维力、敏锐的判断力、充沛的精力,夺取答辩的胜利。(三)保持愉悦的精神状态愉悦的精神状态,_竞选大队长时到台上演讲怎样不紧张端庄大方举止得体
文章浏览阅读2.1k次。select(查询)方法需要集合或者数组去装数据,而update(修改),delete(删除),insert(添加)方法是不需要设置返回类型的,它们都是默认返回一个int。/** * 查询水果 * @return */ @RequestMapping("/selFruit") public List<Fruit> selFruit(){ List<Fruit> list = fruitService...._springboot mapper查詢返回數字
文章浏览阅读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++游戏
文章浏览阅读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如何计算图像
文章浏览阅读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
文章浏览阅读321次,点赞3次,收藏10次。Kivy-iOS: 带着Kivy框架去开发iOS应用[项目链接]简介在移动设备的市场中,Android 和 iOS 是两个主要的操作系统。开发者们通常需要为这两种平台分别开发应用以覆盖更多的用户群。现在有一个叫做 Kivy 的 Python 框架,可以让您在一种语言和一套代码基础上构建跨平台的应用程序,包括了 Android 和 iOS。本文将向您介绍如何利用 Kivy-iOS 工具链帮助您..._kivy怎么调用ios底层硬件
文章浏览阅读196次。参考:http://blog.163.com/xianfuying%40126/blog/static/21960005201181482518631/转载于:https://www.cnblogs.com/z-xx/p/7484416.html_本地仓库连接新的git仓库mac
文章浏览阅读1.9k次。说明百度地图的api还是蛮大方的,反正个人是肯定够用的。没想到还有天气接口,以后也可以玩一玩。本篇实现计算两个地址间的距离。内容1 地址转为经纬度使用这个接口进行地址转换坐标。第一次提交报APP 服务禁用,这时候就进入App的设置,我比较怕麻烦,一次全勾上了。随意输入一个地址address ='上海浦东国际机场卫星厅'---resp = req.get(url_template % (address , ak ))resp.text'showLocation&&s_百度api查2点之间的推荐路线距离
文章浏览阅读4.8w次,点赞23次,收藏102次。如何在word中插入latex公式和伪代码在word中可以通过软件Aurora把latex公式直接搬到word里面,超级好用,安装完软件之后,word的工具栏会自动出现该选项。至于伪代码,只要复制,点击工具栏的 Paste from Tex 就可以直接出现,方便极了!把伪代码搬到word里面可能出现的问题有以下几点:1.主动的算法序号不见了,推荐语句如下,其中星号可以被替代为相应序号\renewco_word插入latex代码块