将安卓app闪退信息保存在本地或服务器的工具类CrashHandler

在我刚刚实习的时候,有一个测试曾经问我,能不能把安卓app闪退时候的错误信息保存在本地文件夹或者上传到服务器。他说这样做可以把客户反映的闪退问题、而且我们公司不能及时重现BUG的时候去使用。当时,我的回答是否定的,因为我当时并不知道原来安卓早就有UncaughtExceptionHandler可以去全局捕获异常,直到我看了Android开发艺术探索这本书。其实写这篇文章不是为了去展示什么,因为书上、网上都有一模一样的继承UncaughtExceptionHandler写一个工具类。写这篇文章我只是想纠正当年的错误并且可以作为自己项目的工具类去使用,毕竟身为程序猿都不喜欢跟“万恶”的客户和“死神”般的测试有过多的冲突摩擦。当他们发现app有闪退问题的时候,你只需要叫他们把app“临终”前保存在本地的文件打开,这里有几个好处:

1.可以轻松的摆脱“万恶”的客户和“死神”般的测试的纠缠,有时候客户和测试不在身边,远距离就可以找到问题所在;
2.你就可以省去很多功夫去重现BUG,可以留下多点时间跟测试的妹子聊天去;
3.可以把国内各大手机商改过的安卓系统无缘无故会在某些型号的手机上出现闪退情况,捕获记录下来,做好适配工作。

如果有很多app的内容要跟测试妹子深入了解的话,你可以把这篇文章省略掉~
好吧,言归正传,UncaughtExceptionHandler是全局捕获异常的,为app意外中止的提供一些处理的方法。用的时候我们只需要实现UncaughtExceptionHandler这个接口,重写一些方法,就可以做到把异常信息保存在本地或者上传到服务器。
代码如下:

public class CrashHandler implements Thread.UncaughtExceptionHandler {    
   private static final String TAG = "CrashHandler";    
   private static final boolean DEBUG = true;   
   private static final String PATH = Environment.getExternalStorageDirectory().getPath() + "/CrashTest/log/"; 
   private static final String FILE_NAME = "crash";  
   private static final String FILE_NAME_SUFFIX = ".txt";  
   private static CrashHandler sInstance = new CrashHandler();    
   private Thread.UncaughtExceptionHandler mDefaultCrashHandler; 
   private Context mContext;  
   private CrashHandler(){ 
   }  
   public static CrashHandler getsInstance(){   
       return sInstance; 
   } 
   public void init(Context context){  
      mDefaultCrashHandler = Thread.getDefaultUncaughtExceptionHandler();    
      Thread.setDefaultUncaughtExceptionHandler(this);   
      mContext = context.getApplicationContext(); 
   }  
  /**  
   * 这个是最关键的函数,当程序中有未被捕获的异常,系统将会自动调用uncaughtException方法  
   * thread为出现未捕获异常的线程,ex为未捕获的异常,有了这个ex,我们就可以得到异常信息   
   * @param thread  
   * @param ex  
   */  
   @Override 
   public void uncaughtException(Thread thread, Throwable ex) { 
       try {     
               //导出异常信息到SD卡中  
              dumpExceptionToSDCard(ex);  
             //这里可以上传异常信息到服务器,便于开发人员分析日志从而解决bug    
             uploadExceptionToServer(); 
           }catch (IOException e){ 
               e.printStackTrace(); 
           }   
          ex.printStackTrace();  
        //如果系统提供默认的异常处理器,则交给系统去结束程序,否则就由自己结束自己
        if(mDefaultCrashHandler != null){  
          mDefaultCrashHandler.uncaughtException(thread, ex);      
        }else { 
           //自己处理  
           try {
               //延迟2秒杀进程   
               Thread.sleep(2000);              
               Toast.makeText(mContext, "程序出错了~", Toast.LENGTH_SHORT).show();   
              } catch (InterruptedException e) {   
             Log.e(TAG, "error : ", e);   
         }    
        android.os.Process.killProcess(Process.myPid()); 
       }  
  }  
  private void dumpExceptionToSDCard(Throwable ex) throws IOException{  
      //如果SD卡不存在或无法使用,则无法把异常信息写入SD卡        
      if(!Environment.getExternalStorageState().equals(Environment.MEDIA_MOUNTED)){  
          if(DEBUG){  
              Log.e(TAG, "sdcard unmounted,skip dump exception");                
              return;  
          }  
      }  
      File dir = new File(PATH);  
      if(!dir.exists()){   
         dir.mkdirs();  
      }  
      long current = System.currentTimeMillis();  
      String time = new SimpleDateFormat("yyyy-MM-dd HH:MM:SS").format(new Date(current));  
      File file = new File(PATH + FILE_NAME + time + FILE_NAME_SUFFIX);  
      try {   
          PrintWriter pw = new PrintWriter(new BufferedWriter(new FileWriter(file)));  
          pw.println(time);  
          dumpPhoneInfo(pw);  
          pw.println();  
          ex.printStackTrace(pw);  
          pw.close();  
          Log.e(TAG, "dump crash info seccess");  
      }catch (Exception e){  
          Log.e(TAG,e.getMessage());  
          Log.e(TAG,"dump crash info failed");  
      }  
  }  
  private void dumpPhoneInfo(PrintWriter pw)throws PackageManager.NameNotFoundException{  
      PackageManager pm = mContext.getPackageManager();  
      PackageInfo pi = pm.getPackageInfo(mContext.getPackageName(),                PackageManager.GET_ACTIVITIES);  
      pw.print("App Version: ");  
      pw.print(pi.versionName);  
      pw.print('_');  
      pw.println(pi.versionCode);  
      //Android版本号  
      pw.print("OS Version: ");  
      pw.print(Build.VERSION.RELEASE);  
      pw.print("_");  
      pw.print(Build.VERSION.SDK_INT);  
      //手机制造商  
      pw.print("Vendor: ");  
      pw.print(Build.MANUFACTURER);  
      //手机型号  
      pw.print("Model: ");  
      pw.println(Build.MODEL);  
      //CPU架构  
      pw.print("CPU ABI: ");  
      pw.println(Build.CPU_ABI);  
  }  
  private void uploadExceptionToServer(){  
      //将异常信息发送到服务器  
  }
}

就是这么简单一个工具类,是不是很简单?好吧下面我们来看看怎么使用~

CrashHandler的使用.png

只需要在application获取CrashHandler的单例和初始化就可以做到全局捕获异常了,跟一些第三方的框架初始化没什么区别吧~
最后看一下效果图:

自己抛出一个异常做测试.png

点击这个按钮,我们就看到app闪退了,然后我们去该保存目录下看看有什么~

测试结果.png

我们可以看到我们还可以把安卓版本、手机型号、运营商输出来,考虑到国内各大手机商改过的安卓系统无缘无故会在某些型号的手机上出现闪退情况,捕获到这些异常还是很有必要的。
好吧,其实以上的代码很大部分都是来源于Android开发艺术探索这本书的,有兴趣的同学也可以去看一下,我只是想告诉更多还不知道的同学,UncaughtExceptionHandler可以把你的闪退信息保存到本地或者上传到服务器上!
源码下载链接

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,723评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,003评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,512评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,825评论 1 290
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,874评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,841评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,812评论 3 416
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,582评论 0 271
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,033评论 1 308
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,309评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,450评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,158评论 5 341
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,789评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,409评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,609评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,440评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,357评论 2 352

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 172,042评论 25 707
  • 文章最早发布于我的微信公众号 Android_De_Home 中,欢迎大家扫描下面二维码关注微信公众获取更多知识...
    sydMobile阅读 4,044评论 0 3
  • 艾瑞巴蒂,又到周五了,静静继续给大家播报新闻,万达酒店要冲出中国了,而今旅(Hotel Jen)也要布局亚洲,哎哟...
    Miss_Raquel阅读 288评论 0 0
  • 做一件事最好的时间是在10年前,其次就是现在,我还总想着等我找到一个女朋友以后就开始学习,这tmd真的是太天真了,...
    现在的孩子真是阅读 126评论 0 0
  • - 地址与值的关系 举例:图中表明了计算机中内存的存在形式,矩形框上面的整数为地址编号(内存完成编号)。这样的编号...
    安公子_阅读 158评论 0 0