> cockroach2.0版,有效减少黑屏

Language

  • English
  • [Chinese]

很多人曲解了这个库的用意,现特声明如下

当APP主线程抛出异常时就会导致APP crash,可能是由于view点击时抛出了异常等等,像这种异常我们更希望即使点击没反应也不要crash,用户顶多会认为是点了没反应,或者认为是本来就不可以点击,这时候就可以使用Cockroach,而且没有其他副作用,用户就跟没点一样,并且不影响其他逻辑。这样总比每次都crash要好很多。当然这个库也存在不确定因素,比如Activity初始化时等抛出了异常,就会导致Activity什么都不显示,但这并不是ANR,是由于Activity生命周期没有执行完整导致,issues中很多人认为这是ANR,进而导致微博上有人说这个库捕获到异常后会导致ANR,其实这个时候主线程并没有被阻塞,也就不存在ANR。当然这个库对于native异常和ANR也是无能为力的,只能保证java异常不会导致crash。

Cockroach

打不死的小强,永不crash的Android。

android 开发中最怕的就是crash,好好的APP测试时没问题,一发布就各种crash,只能通过紧急发布hotfix来解决,但准备hotfix的时间可能很长,导致这段时间用户体验非常差,android中虽然可以通过设置 Thread.setDefaultUncaughtExceptionHandler来捕获所有线程的异常,但主线程抛出异常时仍旧会导致activity闪退,app进程重启。使用Cockroach后就可以保证不管怎样抛异常activity都不会闪退,app进程也不会重启。

推荐使用姿势

  • 当线上发现进入某个Activity有大量crash时,若装载Cockroach后不影响APP运行,不影响用户体检,就可以通过后端控制来自动开启Cockroach,当退出这个Activity后自动卸载Cockroach。这样其他用户再次进入该Activity就不会crash。

  • 可以根据需要在任意地方(不一定要在主线程)装载,在任意地方卸载。可以多次装载和卸载。 例如:

import android.app.Application;
import android.os.Handler;
import android.os.Looper;
import android.util.Log;
import android.widget.Toast;/*** Created by wanjian on 2017/2/14.*/Cockroach.install(new Cockroach.ExceptionHandler() {// handlerException内部建议手动try{  你的异常处理逻辑  }catch(Throwable e){ } ,以防handlerException内部再次抛出异常,导致循环调用handlerException@Overridepublic void handlerException(final Thread thread, final Throwable throwable) {//开发时使用Cockroach可能不容易发现bug,所以建议开发阶段在handlerException中用Toast谈个提示框,//由于handlerException可能运行在非ui线程中,Toast又需要在主线程,所以new了一个new Handler(Looper.getMainLooper()),//所以千万不要在下面的run方法中执行耗时操作,因为run已经运行在了ui线程中。//new Handler(Looper.getMainLooper())只是为了能弹出个toast,并无其他用途new Handler(Looper.getMainLooper()).post(new Runnable() {@Overridepublic void run() {try {//建议使用下面方式在控制台打印异常,这样就可以在Error级别看到红色logLog.e("AndroidRuntime","--->CockroachException:"+thread+"<---",throwable);Toast.makeText(App.this, "Exception Happend\n" + thread + "\n" + throwable.toString(), Toast.LENGTH_SHORT).show();
//                        throw new RuntimeException("..."+(i++));} catch (Throwable e) {}}});}});

卸载 Cockroach

 Cockroach.uninstall();

测试

装载Cockroach后点击view抛出异常和new Handler中抛出异常

   final TextView textView = (TextView) findViewById(R.id.text);findViewById(R.id.install).setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {textView.setText("已安装 Cockroach");install();}});findViewById(R.id.uninstall).setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {textView.setText("已卸载 Cockroach");Cockroach.uninstall();}});findViewById(R.id.but1).setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {throw new RuntimeException("click exception...");}});findViewById(R.id.but2).setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {new Handler().post(new Runnable() {@Overridepublic void run() {throw new RuntimeException("handler exception...");}});}});findViewById(R.id.but3).setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {new Thread() {@Overridepublic void run() {super.run();throw new RuntimeException("new thread exception...");}}.start();}});findViewById(R.id.but4).setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {startActivity(new Intent(getApplicationContext(), SecActivity.class));}});}private void install() {Cockroach.install(new Cockroach.ExceptionHandler() {@Overridepublic void handlerException(final Thread thread, final Throwable throwable) {Log.d("Cockroach", "MainThread: " + Looper.getMainLooper().getThread() + "  curThread: " + Thread.currentThread());new Handler(Looper.getMainLooper()).post(new Runnable() {@Overridepublic void run() {try {Log.e("AndroidRuntime","--->CockroachException:"+thread+"<---",throwable);Toast.makeText(getApplicationContext(), "Exception Happend\n" + thread + "\n" + throwable.toString(), Toast.LENGTH_SHORT).show();
//                        throw new RuntimeException("..."+(i++));} catch (Throwable e) {}}});}});}

捕获到的堆栈如下,可以看到都已经被 at com.wanjian.cockroach.Cockroach$1.run(Cockroach.java:47) 拦截,APP没有任何影响,没有闪退,也没有重启进程

02-16 09:58:00.660 21199-21199/wj.com.fuck E/AndroidRuntime: --->CockroachException:Thread[main,5,main]<---java.lang.RuntimeException: click exception...at wj.com.fuck.MainActivity$3.onClick(MainActivity.java:53)at android.view.View.performClick(View.java:4909)at android.view.View$PerformClick.run(View.java:20390)at android.os.Handler.handleCallback(Handler.java:815)at android.os.Handler.dispatchMessage(Handler.java:104)at android.os.Looper.loop(Looper.java:194)at com.wanjian.cockroach.Cockroach$1.run(Cockroach.java:47)at android.os.Handler.handleCallback(Handler.java:815)at android.os.Handler.dispatchMessage(Handler.java:104)at android.os.Looper.loop(Looper.java:194)at android.app.ActivityThread.main(ActivityThread.java:5826)at java.lang.reflect.Method.invoke(Native Method)at java.lang.reflect.Method.invoke(Method.java:372)at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1009)at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:804)
02-16 09:58:12.401 21199-21199/wj.com.fuck E/AndroidRuntime: --->CockroachException:Thread[main,5,main]<---java.lang.RuntimeException: handler exception...at wj.com.fuck.MainActivity$4$1.run(MainActivity.java:63)at android.os.Handler.handleCallback(Handler.java:815)at android.os.Handler.dispatchMessage(Handler.java:104)at android.os.Looper.loop(Looper.java:194)at com.wanjian.cockroach.Cockroach$1.run(Cockroach.java:47)at android.os.Handler.handleCallback(Handler.java:815)at android.os.Handler.dispatchMessage(Handler.java:104)at android.os.Looper.loop(Looper.java:194)at android.app.ActivityThread.main(ActivityThread.java:5826)at java.lang.reflect.Method.invoke(Native Method)at java.lang.reflect.Method.invoke(Method.java:372)at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1009)at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:804)
02-16 09:58:13.241 21199-21199/wj.com.fuck E/AndroidRuntime: --->CockroachException:Thread[Thread-26326,5,main]<---java.lang.RuntimeException: new thread exception...at wj.com.fuck.MainActivity$5$1.run(MainActivity.java:76)

当卸载Cockroach后再在click中抛出异常,日志如下

02-16 09:59:01.251 21199-21199/wj.com.fuck E/AndroidRuntime: FATAL EXCEPTION: mainProcess: wj.com.fuck, PID: 21199java.lang.RuntimeException: click exception...at wj.com.fuck.MainActivity$3.onClick(MainActivity.java:53)at android.view.View.performClick(View.java:4909)at android.view.View$PerformClick.run(View.java:20390)at android.os.Handler.handleCallback(Handler.java:815)at android.os.Handler.dispatchMessage(Handler.java:104)at android.os.Looper.loop(Looper.java:194)at android.app.ActivityThread.main(ActivityThread.java:5826)at java.lang.reflect.Method.invoke(Native Method)at java.lang.reflect.Method.invoke(Method.java:372)at com.android.internal.os.ZygoteInit$MethodAndArgsCaller.run(ZygoteInit.java:1009)at com.android.internal.os.ZygoteInit.main(ZygoteInit.java:804)

可以看到 at com.wanjian.cockroach.Cockroach$1.run(Cockroach.java:47) 没有拦截,并且APP crash了。

注意

  • 当主线程或子线程抛出异常时都会调用exceptionHandler.handlerException(Thread thread, Throwable throwable)

  • exceptionHandler.handlerException可能运行在非UI线程中。

  • handlerException内部建议手动try{ 你的异常处理逻辑 }catch(Throwable e){ } ,以防handlerException内部再次抛出异常,导致循环调用handlerException

  • 若设置了Thread.setDefaultUncaughtExceptionHandler则可能无法捕获子线程异常。

  • 最佳拍档android.arch.lifecycle.LiveData+Cockroach。当使用LiveData.postValue时,Observer会在一个单独的消息中执行,这时 若Observer中发生了异常,就可以被cockroach捕获到,不会有其他影响。

虽然可以捕获到所有异常,但可能会导致一些莫名其妙的问题,比如view初始化时发生了异常,异常后面的代码得不到执行,虽然不 会导致app crash但view内部已经出现了问题,运行时就会出现很奇葩的现象。再比如activity声明周期方法中抛出了异常,则生 命周期就会不完整,从而导致各种奇葩的现象。

Cockroach采用android标准API编写,无依赖,足够轻量,轻量到只有不到100行代码,一般不会存在兼容性问题,也不存在性能上的问题,可以兼容所有android版本。

已上传到jcenter, compile 'com.wanjian:cockroach:0.0.5'

效果视频 https://github.com/android-notes/Cockroach/blob/master/cockroach.mp4?raw=true

原理分析

原理分析

Android杜绝崩溃降低Android非必要crash相关推荐

  1. android 杜绝崩溃闪退处理

    当APP主线程抛出异常时就会导致APP crash,可能是由于view点击时抛出了异常等等,像这种异常我们更希望即使点击没反应也不要crash,用户顶多会认为是点了没反应,或者认为是本来就不可以点击, ...

  2. android 程序崩溃处理,Android应用崩溃的应急处理

    在我们开发Android应用程序的时候总是难免遇到程序崩溃的问题:(很抱歉,"××"已停止运行.) 每次看到这个问题心都是哇凉哇凉的 一般遇到这样问题,有两种可能: 1.自己的代码 ...

  3. android 保存崩溃信息,Android保存每次运行崩溃报告的日志

    您可以通过以下链接找到帮助 Writing crash reports into device sd card 您不需要添加外部库. import com.wordpress.doandroid.Tr ...

  4. android随机崩溃莫名其妙,Android CrashHandler编写自己的异常捕获的方法

    平时写代码,我们可能会抛出各种异常,这些异常有些是我们测试过程中发现进行解决的,但是也有一些异常是我们未知的,不论是代码的逻辑问题还是Android本身底层的一些bug,我们都需要及时了解并进行解决. ...

  5. android xlog崩溃日志,Android第三方log库:xlog使用记录

    第一步:由于xlog发布在jitpack 仓库,所以如果android studio没有在project目下的的build.gradle下配置jitpack仓库的话需要配置下 allprojects ...

  6. Android异常崩溃Crash重启方案

    在Android开发过程中,如果有Crash,我们往往想看到具体Crash的情景,但是在发布版本后,应用万一出现崩溃现象,就会出现一个弹窗说应用崩溃了,如果给用户看到,会有很大不良印象,如果是我,我觉 ...

  7. android的崩溃报告(crash dump)

    ACRA (Application Crash Report For Android)就是为了解决android 应用程序崩溃以后的发送报告的问题.  你能在https://github.com/AC ...

  8. android 程序崩溃后自动重启,【Android】App应用崩溃(Crash/Force Close)之后如何让它自动重启?...

    英文原文: Auto Restart application after Crash/Force Close in Android. 手机上的Android应用,经常会出现"Force Cl ...

  9. Android防止App崩溃的,Android App Crash之后如何禁止Activity重启

    针对crash,android默认的处理方式是,退出app.弹一个提示框. /** * 初始化 * * @param context */ public void init(context conte ...

最新文章

  1. flac3d命令流实例大全_ANSYS APDL 疲劳分析实例附命令流
  2. 敏捷开发用户故事系列之二:如何面向客户价值编写故事
  3. 对CSS了解-overflow:hidden
  4. 深度学习在搜索和推荐领域的应用
  5. python typeerror* wants int_python-TypeError:’int’对象是不可迭代的?
  6. kafka 发布-订阅模式_使用Apache Kafka作为消息系统的发布-订阅通信中的微服务,并通过集成测试进行了验证...
  7. 倒N字形排列java_Java排序8大算法实现
  8. CF429E Points and Segments
  9. 洛谷 1315 观光公交——贪心
  10. 利用python处理excel文件并可视化处理-python如何将excel数据处理可视化
  11. UNIX时间戳和北京时间的相互转换
  12. 【网站介绍】有意思的网站(备忘)
  13. 采集PMS7003 颗粒度传感器数据,显示在TFT屏幕上,并通过webapi上传(1/2)
  14. 视频管理系统源码php,video cms 视频分享管理系统
  15. 51页大数据湖总体规划及大数据湖一体化运营管理方案
  16. 云计算平台建设总体技术方案
  17. 光通信的最新技术趋势
  18. 芯片内部长啥样?牛人用1500张照片,一层层放给你看
  19. 泛微协同OA房地产行业解决方案
  20. c语言怎么打出花边图案,甄嬛舌尖各种版本苏牙咬人 连黄梅戏C语言都来了

热门文章

  1. dlib 怎么安装vs2017_dlib开发环境部署
  2. 简单认识Windows图标
  3. python hook pc微信_微信机器人之PC微信hook
  4. 思科再发两款新品 将IBN能力应用到广域网
  5. nginx-rtmp(直播点播)配置
  6. 0505|713. 乘积小于 K 的子数组
  7. SCILAB及其在控制系统仿真中的应用
  8. 英博啤酒玩转区域品牌 ---陈标杰
  9. SMK电路板USBType-C插座CSS5324-3M01F CSS5324-0M02F连接器以及过压保护
  10. 2021年云南高考成绩在哪里查询,云南教育考试院:2021年云南高考成绩查询入口(三种方式查分)...