Toast在子线程调用的问题

作者: df556ada620a | 来源:发表于2018-12-05 20:17 被阅读35次

    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进阶技术,面试资料系统整理分享,职业生涯规划,产品,思维,行业观察,谈天说地。可以加Android架构师群;701740775。

    相关文章

      网友评论

        本文标题:Toast在子线程调用的问题

        本文链接:https://www.haomeiwen.com/subject/gptgcqtx.html