Toast 在 Android 7.1 崩潰排查及修復(fù)

崩潰詳情

嘗試復(fù)現(xiàn)

  • 通過崩潰信息從網(wǎng)上找到的一些論述著洼,發(fā)現(xiàn)這個問題是因?yàn)橹骶€程被阻塞了樟遣,而 Toast 沒有及時銷毀導(dǎo)致的,那么接下來讓我們對它進(jìn)行復(fù)現(xiàn)
  • 為什么出現(xiàn)這個問題身笤,是因?yàn)?Toast 的顯示是通過 Handler.sendMessage豹悬,所以這個操作是異步的,而 Thread.sleep 會阻塞主線程液荸,從而導(dǎo)致 Handler.handleMessage 在接收到消息的時候 WindowToken 已經(jīng)失效了

  • 經(jīng)過實(shí)際的測試:如果是短吐司瞻佛,sleep 2000 毫秒的時候還是會拋出異常,sleep 1500 毫秒則不會發(fā)生異常娇钱;如果是長吐司伤柄,sleep 3500 毫秒的時候也是會拋出異常,sleep 3000 毫秒的時候就不會發(fā)生異常

  • 由此可見忍弛,WindowToken 失效的時間是跟 Toast 的顯示時長有關(guān)响迂,如果是短吐司,那么 WindowToken 有效時長只能在 2 秒以內(nèi)细疚;如果是長吐司蔗彤,那么 WindowToken 的有效時長只能在 3.5 秒以內(nèi)

  • 然后再通過 WindowManager.addView 的時候,它會對 WindowToken 例行檢查疯兼,如果是失效狀態(tài)則會拋出異常給上層然遏,而這個機(jī)制恰好是 Android 7.1 的時候才有的,谷歌那個時候并沒有考慮到對 Toast 的一些處理吧彪。因?yàn)橥ㄟ^瀏覽 Android 7.0 和 Android 6.0 的源碼待侵,發(fā)現(xiàn)谷歌也是沒有進(jìn)行 try 處理,但是崩潰的機(jī)型卻全是清一色的 Android 7.1

問題排查

  • 通過查看這個崩潰都是在 Android 7.1 的機(jī)型才會出現(xiàn)姨裸,那么我們可以對比 Android 7.1 的源碼和 Android 8.0 看看
  • 通過追蹤不同 API 等級的源碼秧倾,發(fā)現(xiàn)這個問題在 Android 8.0 上面已經(jīng)被被修復(fù)了

  • 通過查看 Toast 的源碼,發(fā)現(xiàn) Toast 其實(shí)就是一個 WindowManager傀缩,并且通過 Handler 來顯示和隱藏那先。

  • 而產(chǎn)生崩潰的地方是在 handleShow 方法里面

  • 而 handleShow 方法是被 Toast 中的名為 TN 靜態(tài)內(nèi)部類中的 Handler 對象調(diào)用

進(jìn)行修復(fù)

  • 那么解決這一問題的方式的思路是,將這個 Handler 對象通過反射獲取到赡艰,然后使用靜態(tài)代理的方式對它進(jìn)行回調(diào)并對進(jìn)行捕獲異常
  • 最后經(jīng)過驗(yàn)證售淡,是 OK 的,已經(jīng)沒有崩潰的問題出現(xiàn)了。

  • 但是新的問題又出現(xiàn)了揖闸,我們以前寫 Toast 是這樣的

Toast.makeText(this, "666", Toast.LENGTH_LONG).show();
  • 但是如果為了修復(fù)這個崩潰問題揍堕,我們需要這樣寫
Toast toast = Toast.makeText(this, "666", Toast.LENGTH_LONG);
if (Build.VERSION.SDK_INT == Build.VERSION_CODES.N_MR1) {
    try {

        // 獲取 mTN 字段對象
        Field mTNField = Toast.class.getDeclaredField("mTN");
        mTNField.setAccessible(true);
        Object mTN = mTNField.get(toast);

        // 獲取 mTN 中的 mHandler 字段對象
        Field mHandlerField = mTNField.getType().getDeclaredField("mHandler");
        mHandlerField.setAccessible(true);
        final Handler mHandler = (Handler) mHandlerField.get(mTN);

        // 偷梁換柱
        mHandlerField.set(mTN, new Handler() {
            @Override
            public void handleMessage(Message msg) {
                // 捕獲這個異常,避免程序崩潰
                try {
                    mHandler.handleMessage(msg);
                } catch (WindowManager.BadTokenException ignored) {}
            }
        });

    } catch (IllegalAccessException | NoSuchFieldException ignored) {}
}

toast.show();
  • 這樣寫感覺心好累汤纸,不想這樣寫衩茸,有沒有一種方式可以一勞永逸?

  • 答案當(dāng)然是有了蹲嚣,使用第三方 Toast 封裝的框架:https://github.com/getActivity/Toaster递瑰,框架內(nèi)部已經(jīng)處理了這個問題,調(diào)用者無需關(guān)心此問題隙畜。

  • 使用框架后抖部,可以這么寫

Toaster.show("666");
  • 還是一句代碼,就問你 6 不 6

問題總結(jié)

  • 問題描述:Toast 在主線程阻塞情況下會導(dǎo)致 WindowToken 失效议惰,從而導(dǎo)致應(yīng)用崩潰

  • 涉及范圍:所有 Android 版本為 7.1 的用戶慎颗,并且項(xiàng)目中使用了原生 Toast 的地方都有可能觸發(fā)崩潰

  • 解決方案:不直接使用原生 Toast,而使用第三方 Toast 框架

Android 技術(shù)討論 Q 群:10047167

最后編輯于
?著作權(quán)歸作者所有,轉(zhuǎn)載或內(nèi)容合作請聯(lián)系作者
  • 序言:七十年代末言询,一起剝皮案震驚了整個濱河市俯萎,隨后出現(xiàn)的幾起案子,更是在濱河造成了極大的恐慌运杭,老刑警劉巖夫啊,帶你破解...
    沈念sama閱讀 216,324評論 6 498
  • 序言:濱河連續(xù)發(fā)生了三起死亡事件,死亡現(xiàn)場離奇詭異辆憔,居然都是意外死亡撇眯,警方通過查閱死者的電腦和手機(jī),發(fā)現(xiàn)死者居然都...
    沈念sama閱讀 92,356評論 3 392
  • 文/潘曉璐 我一進(jìn)店門虱咧,熙熙樓的掌柜王于貴愁眉苦臉地迎上來熊榛,“玉大人,你說我怎么就攤上這事腕巡⌒梗” “怎么了?”我有些...
    開封第一講書人閱讀 162,328評論 0 353
  • 文/不壞的土叔 我叫張陵绘沉,是天一觀的道長煎楣。 經(jīng)常有香客問我,道長车伞,這世上最難降的妖魔是什么择懂? 我笑而不...
    開封第一講書人閱讀 58,147評論 1 292
  • 正文 為了忘掉前任,我火速辦了婚禮帖世,結(jié)果婚禮上,老公的妹妹穿的比我還像新娘。我一直安慰自己日矫,他們只是感情好赂弓,可當(dāng)我...
    茶點(diǎn)故事閱讀 67,160評論 6 388
  • 文/花漫 我一把揭開白布。 她就那樣靜靜地躺著哪轿,像睡著了一般盈魁。 火紅的嫁衣襯著肌膚如雪。 梳的紋絲不亂的頭發(fā)上窃诉,一...
    開封第一講書人閱讀 51,115評論 1 296
  • 那天杨耙,我揣著相機(jī)與錄音,去河邊找鬼飘痛。 笑死珊膜,一個胖子當(dāng)著我的面吹牛,可吹牛的內(nèi)容都是我干的宣脉。 我是一名探鬼主播车柠,決...
    沈念sama閱讀 40,025評論 3 417
  • 文/蒼蘭香墨 我猛地睜開眼,長吁一口氣:“原來是場噩夢啊……” “哼塑猖!你這毒婦竟也來了竹祷?” 一聲冷哼從身側(cè)響起,我...
    開封第一講書人閱讀 38,867評論 0 274
  • 序言:老撾萬榮一對情侶失蹤羊苟,失蹤者是張志新(化名)和其女友劉穎塑陵,沒想到半個月后,有當(dāng)?shù)厝嗽跇淞掷锇l(fā)現(xiàn)了一具尸體蜡励,經(jīng)...
    沈念sama閱讀 45,307評論 1 310
  • 正文 獨(dú)居荒郊野嶺守林人離奇死亡令花,尸身上長有42處帶血的膿包…… 初始之章·張勛 以下內(nèi)容為張勛視角 年9月15日...
    茶點(diǎn)故事閱讀 37,528評論 2 332
  • 正文 我和宋清朗相戀三年,在試婚紗的時候發(fā)現(xiàn)自己被綠了巍虫。 大學(xué)時的朋友給我發(fā)了我未婚夫和他白月光在一起吃飯的照片彭则。...
    茶點(diǎn)故事閱讀 39,688評論 1 348
  • 序言:一個原本活蹦亂跳的男人離奇死亡,死狀恐怖占遥,靈堂內(nèi)的尸體忽然破棺而出俯抖,到底是詐尸還是另有隱情,我是刑警寧澤瓦胎,帶...
    沈念sama閱讀 35,409評論 5 343
  • 正文 年R本政府宣布芬萍,位于F島的核電站,受9級特大地震影響搔啊,放射性物質(zhì)發(fā)生泄漏柬祠。R本人自食惡果不足惜,卻給世界環(huán)境...
    茶點(diǎn)故事閱讀 41,001評論 3 325
  • 文/蒙蒙 一负芋、第九天 我趴在偏房一處隱蔽的房頂上張望漫蛔。 院中可真熱鬧,春花似錦、人聲如沸莽龟。這莊子的主人今日做“春日...
    開封第一講書人閱讀 31,657評論 0 22
  • 文/蒼蘭香墨 我抬頭看了看天上的太陽毯盈。三九已至剃毒,卻和暖如春,著一層夾襖步出監(jiān)牢的瞬間搂赋,已是汗流浹背赘阀。 一陣腳步聲響...
    開封第一講書人閱讀 32,811評論 1 268
  • 我被黑心中介騙來泰國打工, 沒想到剛下飛機(jī)就差點(diǎn)兒被人妖公主榨干…… 1. 我叫王不留脑奠,地道東北人基公。 一個月前我還...
    沈念sama閱讀 47,685評論 2 368
  • 正文 我出身青樓,卻偏偏與公主長得像捺信,于是被迫代替她去往敵國和親酌媒。 傳聞我的和親對象是個殘疾皇子,可洞房花燭夜當(dāng)晚...
    茶點(diǎn)故事閱讀 44,573評論 2 353

推薦閱讀更多精彩內(nèi)容