> 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
@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+"<---",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() {
@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+"<---",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: 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則可能無法捕獲子線程異常。
-
最佳拍檔
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