Toast在子线程调用的问题

Toast我们平时经常使用,但是你是否了解在子线程中要如何使用Toast呢?

Toast的一般姿势

平时我们经常在主线程中直接使用Toast,代码看起来会像下面这样

Toast.makeText(MainActivity.this, "", Toast.LENGTH_SHORT).show();

但是如果在子线程调用是不会有toast弹出的

Toast的正确姿势

如果在子线程调用那么让Toast能正常显示的方式是在它之前和之后调用Looper.prepare()和Looper.loop()

Looper.prepare();
Toast.makeText(MainActivity.this, "", Toast.LENGTH_SHORT).show();
Looper.loop();

原因是什么呢

我们得从源码角度来分析,看看在Toast show()的时候做了些什么

public void show() {
    if (mNextView == null) {
        throw new RuntimeException("setView must have been called");
    }

    INotificationManager service = getService();
    String pkg = mContext.getOpPackageName();
    TN tn = mTN;
    tn.mNextView = mNextView;

    try {
        service.enqueueToast(pkg, tn, mDuration);
    } catch (RemoteException e) {
        // Empty
    }
}

所以Toast其实是通过NotificationManagerService来实现Toast的展示的,而传给他的参数里的 mTn又是什么呢, 其实它是Toast的一个内部类,它有两个方法,show()和hide()是用来给NotificationManagerService回调的,可以看看它的代码

private static class TN extends ITransientNotification.Stub {
    ....
    /**
     * schedule handleShow into the right thread
     */
    @Override
    public void show(IBinder windowToken) {
        if (localLOGV) Log.v(TAG, "SHOW: " + this);
        mHandler.obtainMessage(SHOW, windowToken).sendToTarget();
    }

    /**
     * schedule handleHide into the right thread
     */
    @Override
    public void hide() {
        if (localLOGV) Log.v(TAG, "HIDE: " + this);
        mHandler.obtainMessage(HIDE).sendToTarget();
    }

因此可以看出来,Toast通过 NotificationManagerService来统一调度 Toast,而 NotificationManagerService回调 TN 的show()来往对应的线程发消息,

既然是handler实现,那么来看看它的实现代码,就在TN的构造方法里有这么一段

if (looper == null) {
    // Use Looper.myLooper() if looper is not specified.
    looper = Looper.myLooper();
    if (looper == null) {
        throw new RuntimeException(
                "Can't toast on a thread that has not called Looper.prepare()");
    }
}
mHandler = new Handler(looper, null) {....

因此没有调用prepare()和启动消息队列的话,在子线程调用Toast是显示不出来的。

总结

Toast在主线程的显示只需要调用show()就可以,如果想在子线程调用,则需要在子线程启动Looper,这样才能有消息队列来承载Handler收发消息。否则子线程的Toast是不能显示的

原文发布于微信公众号 - Android每日一讲(gh_f053f29083b9)

原文发表时间:2018-03-16

本文参与腾讯云自媒体分享计划,欢迎正在阅读的你也加入,一起分享。

发表于

我来说两句

0 条评论
登录 后参与评论

相关文章

扫码关注云+社区

领取腾讯云代金券