Language

[Chinese]

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

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

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

下文也明确说明了

可以根据需要在任意地方装载,在任意地方卸载。

虽然可以捕获到所有异常,但可能会导致一些莫名其妙的问题,比如view初始化时发生了异常,异常后面的代码得不到执行,虽然不

会导致app crash但view内部已经出现了问题,运行时就会出现很奇葩的现象。再比如activity声明周期方法中抛出了异常,则生

命周期就会不完整,从而导致各种奇葩的现象。

所以关键是要如何正确利用这个库

Cockroach

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

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

关于DefaultUncaughtExceptionHandler的用法参考这 DefaultUncaughtExceptionHandler

使用方式

自定义Application继承自android的Application,并在Application中装载,越早初始化越好,可以在Aplication的onCreate中初始化,当然也可以根据需要在任意地方(不一定要在主线程)装载,在任意地方卸载。可以多次装载和卸载。

例如:

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.

*/

public class App extends Application {

@Override

public void onCreate() {

super.onCreate();

Cockroach.install(new Cockroach.ExceptionHandler() {

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

@Override

public 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() {

@Override

public void run() {

try {

//建议使用下面方式在控制台打印异常,这样就可以在Error级别看到红色log

Log.e("AndroidRuntime","--->CockroachException:"+thread+"

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() {

@Override

public void onClick(View v) {

textView.setText("已安装 Cockroach");

install();

}

});

findViewById(R.id.uninstall).setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

textView.setText("已卸载 Cockroach");

Cockroach.uninstall();

}

});

findViewById(R.id.but1).setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

throw new RuntimeException("click exception...");

}

});

findViewById(R.id.but2).setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

new Handler().post(new Runnable() {

@Override

public void run() {

throw new RuntimeException("handler exception...");

}

});

}

});

findViewById(R.id.but3).setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

new Thread() {

@Override

public void run() {

super.run();

throw new RuntimeException("new thread exception...");

}

}.start();

}

});

findViewById(R.id.but4).setOnClickListener(new View.OnClickListener() {

@Override

public void onClick(View v) {

startActivity(new Intent(getApplicationContext(), SecActivity.class));

}

});

}

private void install() {

Cockroach.install(new Cockroach.ExceptionHandler() {

@Override

public 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() {

@Override

public void run() {

try {

Log.e("AndroidRuntime","--->CockroachException:"+thread+"

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: main

Process: wj.com.fuck, PID: 21199

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 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则可能无法捕获子线程异常。

虽然可以捕获到所有异常,但可能会导致一些莫名其妙的问题,比如view初始化时发生了异常,异常后面的代码得不到执行,虽然不

会导致app crash但view内部已经出现了问题,运行时就会出现很奇葩的现象。再比如activity声明周期方法中抛出了异常,则生

命周期就会不完整,从而导致各种奇葩的现象。

虽然会导致各种奇葩问题发生,但可以最大程度的保证APP正常运行,很多时候我们希望主线程即使抛出异常也不影响app的正常使用,比如我们

给某个view设置背景色时,由于view是null就会导致app crash,像这种问题我们更希望即使view没法设置颜色也不要crash,这

时Cockroach就可以满足你的需求。

handlerException(final Thread thread, final Throwable throwable)内部建议请求自己服务器决定该如何处理该异常,是

直接忽略还是杀死APP又或者其他操作。

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

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

原理分析

android永不闪退版,Cockroach相关推荐

  1. 解决delphi10.2.3 android tools闪退

    解决delphi10.2.3 android tools闪退 修改D:\Program Files (x86)\Embarcadero\Studio\19.0\PlatformSDKs\android ...

  2. App测试Android的闪退总结

    Android的闪退有三种情况: 第一种:没有任何弹框提示,直接退出 第二种:有弹框提示程序异常 第三种:ANR无响应 三种情况的日志提交和检索方法:  第一种情况:是底层C挂了 **首先:提交客户端 ...

  3. android 10闪退如何解决,安卓Activity启动在10.0下闪退

    锐英源精品开源,禁止转载和任何形式的非法内容使用,违者必究 安卓Activity启动在10.0下闪退 背景 近期开发的安卓App在数据量大时,安卓10.0下启动Activity时闪退,安卓8.0下不闪 ...

  4. android 定位 闪退_Android使用百度地图出现闪退及定位时显示蓝屏问题

    1.Android使用百度地图出现闪退 一般情况下出现闪退是在AndroidManifest.xml文件中未在application标签中配置 但是,有些同学会发现,所有配置都是按照官网或者教程上的步 ...

  5. android socket 闪退,android客户端闪退

    老师,我把客户端的代码放在android程序里,给一个button监听事件,就是调用这个socket方法,但是一点button就闪退了,没连接上~protected void onCreate(Bun ...

  6. android弹窗闪退,[Android 小记] PopupWindow中放置Spinner 点击直接闪退的问题分析

    错误原因:android.view.WindowManager$BadTokenException: Unable to add window -- token android.view.ViewRo ...

  7. android半透明闪退,(Android)react-native-splash-screen实践-解决react-native打包好后启动白屏的问题...

    1.安装 npm i react-native-splash-screen --save or yarn add react-native-splash-screen --save 2.自动配置 re ...

  8. android半透明闪退,移动端常见面试题一:移动端兼容解决方案

    原标题:移动端常见面试题一:移动端兼容解决方案 移动端 经常出现的兼容问题 1.安卓浏览器看背景图片,有些设备会模糊 因为手机分辨率太小,如果按照分辨率来显示网页,字会非常小,安卓手机devicePi ...

  9. android 定位 闪退_Unity3D研究院之全方位定位Android闪退(九十三)

    首先强烈建议大家安卓打包一定要用il2cpp,我们项目对比了mono发现il2cpp闪退方面比mono少了很多.平常给QA打包推荐勾选Development Build 和 Autocontent P ...

最新文章

  1. python 如何将字符串列表合并后转换成字符串? ''.join(List(str))函数
  2. 四种连接数据库的方法(DriverManager、DataSource子类、DBCP、c3p0)
  3. 带有静态方法的类(java中的math类)
  4. 第八节:Task的各类TaskTResult返回值以及通用线程的异常处理方案
  5. linux下qq怎么截图,ubuntu 12.04使用QQ截图安装教程
  6. 第二章 Flask——Flask中的request
  7. Lync Server 2010 权限相关
  8. 【数据结构笔记05】堆栈及其顺序存储、链式存储
  9. 插件Sytrus合成器功能介绍
  10. python爬虫——爬取taptap游戏的评论信息(通过fiddler抓包)
  11. 拦截图片代码 精易web浏览器_精易Web浏览器 UChk验证源码
  12. 实现微信小程序授权获取手机号登录(c#后端代码附上 少爷接收)
  13. 零基础小白如何系统的自学网络安全(包含学习路线、工作方向)
  14. 《网络攻防》第二周作业
  15. java项目集成谷歌登录验证器,给网站添加谷歌验证(我不是机器人)
  16. 注册mysql确认邮件_如何在MySQL中使用电子邮件ID选择@之前的所有内容?
  17. 无人机航测流程详解:航线规划、像控点布设、CC刺点建模及CASS成图
  18. [转] Android Miracast投屏入门
  19. 几何线条科技感员工转正述职报告PPT模板-优页文档
  20. 慧荣SM2246主控固态硬盘开卡一直pretest解决方法

热门文章

  1. 基于VBA的数据录入界面开发
  2. verdi怎么调大字体
  3. Java基础篇(二)
  4. @sun.misc.Contended 伪共享
  5. 【图像分割】基于模糊C均值聚类算法CGFFCM实现彩色图像分割附matlab代码
  6. Vuetify下载图片的问题探究
  7. 一、求相关函数 corrcoef+协方差矩阵cov
  8. python3字典详解_Python3中Dictionary(字典)操作详解
  9. 实验三 中国剩余定理
  10. C语言中exit(0) 、exit(1) 和return的区别