Toast在子线程调用的问题

691 阅读2分钟
原文链接: mp.weixin.qq.com

欢迎推荐人才赚取三位数以上的佣金,活动规则,只要邀请被推荐人关注公众号点击"面试内推",同时回复他的ID给小A,小A会在被推荐人成功入职后跟你分享佣金。佣金有多少呢?文末有惊喜!

下面进入今天的正题。

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是不能显示的

请随意转发下图到微信群或朋友圈,内推成功的同学,小A将与你分享4位数的佣金哦,最低也有1000大洋!

想了解更多面试知识?请点击阅读原文!