美文网首页Android开发Android进阶之路
面试官问:Android中子线程为什么不能更新UI,该怎么答?

面试官问:Android中子线程为什么不能更新UI,该怎么答?

作者: Android进阶小麦 | 来源:发表于2020-05-29 20:12 被阅读0次

    天才少年_来到一家公司等待面试中。。。
    一个眼睛又大又亮的小姐姐,萌萌的站在我去 的面前。 你像一片轻柔的云在我眼前飘来飘去,你清丽秀雅的脸上荡漾着春天般美丽的笑容,我连我们孩子的名字都起好了。等等,我tm不是来面试的吗?

    小伙子,听说你是来面试的,我是今天的面试官,你先介绍一下你自己吧。

    我叫【天才少年_】,男,30未婚,家里有车有房,我的优点是英俊潇洒,我的座右铭是:既往不纠结,纵情向前看,继续努力。

    额,你这介绍,怎么感觉是来相亲的。

    果然面试官已经被我英俊的外表深深吸引,不能自拔,嗯,萌萌的外表都是不太聪明的样子,今天面试有希望啦,我心中一阵暗喜。

    Android消息处理机制(Handler、Looper、MessageQueue与Message)已经被问烂了,那我们今天来谈谈为什么需要主线程更新UI,子线程不能更新UI?

    卧槽,不按套路出牌啊,果然漂亮的女人都难搞定。

    1)首先,并非在子线程里面更新UI就一定有问题,如下所示的代码,则可以完美更新UI。

     @Override
        public void onActivityCreated(Bundle savedInstanceState) {
            super.onActivityCreated(savedInstanceState);
            init();
            new Thread(new Runnable() {
                @Override
                public void run() {
                    tv_sport_mile.setText("测试界面更新");
                }
            }).start();
        }
    

    但是,如果我们让线程等待2秒后再更新UI,则会发生报错,代码如下所示:

     @Override
        public void onActivityCreated(Bundle savedInstanceState) {
            super.onActivityCreated(savedInstanceState);
            init();
            new Thread(new Runnable() {
                @Override
                public void run() {
                    try {
                        Thread.sleep(2000);
                    } catch (InterruptedException e) {
                        e.printStackTrace();
                    }
                    tv_sport_mile.setText("测试界面更新");
                }
            }).start();
        }
    

    异常报错日志如下图所示:

     android.view.ViewRootImpl$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.
            at android.view.ViewRootImpl.checkThread(ViewRootImpl.java:7021)
            at android.view.ViewRootImpl.requestLayout(ViewRootImpl.java:1047)
    
    在这里插入图片描述

    <figcaption></figcaption>

    为什么在onActivityCreated方法里面可以实现子线程更新UI,但是线程等待两秒后就异常呢?

    你要是不傻,你就知道,肯定是刷新线程判断时机的原因,当时这是我的心理想法,脑子里说不要,嘴上还是很真诚的。

    从at android.view.ViewRootImpl.checkThread(ViewRootImpl.java:7021)的报错可以看到是在ViewRootIml类的checkThread方法中出现异常,多说无益,开启撸源码:

    [图片上传失败...(image-f474fe-1590753965688)]

    <figcaption></figcaption>

    我们首先看ViewRootImpl源码中的requestLayout()和checkThread()方法:

        @Override
        public void requestLayout() {
            if (!mHandlingLayoutInLayoutRequest) {
                checkThread();
                mLayoutRequested = true;
                scheduleTraversals();
            }
        }
    

    view的绘制流程是从scheduleTraversals()方法开始的,包括很多面试官喜欢问的onMeasure、onLayout、onDraw都是由该方法发起的。而在调用scheduleTraversals()方法前,调用了checkThread()方法,该方法会检查当前线程是否跟VewiRootImpl的线程一致,因为VewiRootImpl一般都是在主线程中创建,所以一般都说为是否为主线程。

     void checkThread() {
            if (mThread != Thread.currentThread()) {
                throw new CalledFromWrongThreadException(
                        "Only the original thread that created a view hierarchy can touch its views.");
            }
        }
    

    如果当前线程不是主线程,则抛出异常Only the original thread that created a view hierarchy can touch its views,跟我们的异常一直吻合。总结一下就是在刷新页面前会判断当前是否在主线程,如果不在主线程则抛异常,所以我们开始学Android的时候,别人就告诉我们:更新UI一定要在主线程。

    那为什么上面第一次没有线程等待的时候没有报错呢?可以讲讲吗?

    我想...大概,可能是ViewRootImp还没有创建出来吧,所以没有走到checkThread()方法。

    ViewRootImp什么时候创建的,在onActivityCreated方法后面吗?

    我想起了那个风黑夜高的晚上,我跟小韩(我们部门的程序媛)干着羞羞的事情,嘿嘿~~ 不对,是一起加班看源码的经历,我努力回忆着ViewRootImp的创建过程。

    从ActivityThread源码开始,找到handleResumeActivity()方法:

     final void handleResumeActivity(IBinder token,
                boolean clearHide, boolean isForward, boolean reallyResume, int seq, String reason) {
           ...
            mSomeActivitiesChanged = true;
    
            // TODO Push resumeArgs into the activity for consideration
            r = performResumeActivity(token, clearHide, reason);
    
            if (r != null) {
                final Activity a = r.activity;
    
                if (localLOGV) Slog.v(
                    TAG, "Resume " + r + " started activity: " +
                    a.mStartedActivity + ", hideForNow: " + r.hideForNow
                    + ", finished: " + a.mFinished);
    
                final int forwardBit = isForward ?
                        WindowManager.LayoutParams.SOFT_INPUT_IS_FORWARD_NAVIGATION : 0;
    
                // If the window hasn't yet been added to the window manager,
                // and this guy didn't finish itself or start another activity,
                // then go ahead and add the window.
                boolean willBeVisible = !a.mStartedActivity;
                if (!willBeVisible) {
                    try {
                        willBeVisible = ActivityManager.getService().willActivityBeVisible(
                                a.getActivityToken());
                    } catch (RemoteException e) {
                        throw e.rethrowFromSystemServer();
                    }
                }
                ...
                    r.activity.mVisibleFromServer = true;
                    mNumVisibleActivities++;
                    if (r.activity.mVisibleFromClient) {
                        r.activity.makeVisible();
                    }
                }
                ...
            }
        }
    

    从上面的代码可以看到,调用r.activity.makeVisible();我们看下Activity的makeVisible()的处理逻辑

    void makeVisible() {
            if (!mWindowAdded) {
                ViewManager wm = getWindowManager();
                wm.addView(mDecor, getWindow().getAttributes());
                mWindowAdded = true;
            }
            mDecor.setVisibility(View.VISIBLE);
        }
    

    通过上面的方法可以看到,makeVisible调用了WindowManager的addView方法,WindowManager是个接口,他的具体实现类是WindowManagerImp,直接看WindowManagerImp的addView()方法:

        @Override
        public void addView(@NonNull View view, @NonNull ViewGroup.LayoutParams params) {
            applyDefaultToken(params);
            mGlobal.addView(view, params, mContext.getDisplay(), mParentWindow);
        }
    

    mGlobal是WindowManagerGlobal对象,即调用了WindowManagerGlobal的addView方法,继续深入,快乐继续。

     public void addView(View view, ViewGroup.LayoutParams params,
                Display display, Window parentWindow) {
           ...
                root = new ViewRootImpl(view.getContext(), display);
    
                view.setLayoutParams(wparams);
    
                mViews.add(view);
                mRoots.add(root);
                mParams.add(wparams);
    
                // do this last because it fires off messages to start doing things
                try {
                    root.setView(view, wparams, panelParentView);
                } catch (RuntimeException e) {
                    // BadTokenException or InvalidDisplayException, clean up.
                    if (index >= 0) {
                        removeViewLocked(index, true);
                    }
                    throw e;
                }
            }
        }
    

    这边可以看到创建ViewRootImpl对象,后面View的刷新正是通过ViewRootImpl实现的,由于你面试官没有问,这边不展开讨论,不然把我留到天黑,面试官可能有危险,嘿嘿。

    赠送一个知识点:真正把mDecor加到WindowManager上是并显示出来在makeVisible()方法中实现的,Activity的Window才能正在被使用。

    小伙子理解讲得还不错哦 那ViewRootImp是在onActivityCreated方法后面创建的吗?

    看来面试官小姐姐还是没有忘记这个问题,我们回过头来看handleResumeActivity()

     final void handleResumeActivity(IBinder token,
                boolean clearHide, boolean isForward, boolean reallyResume, int seq, String reason) {
           ...
            mSomeActivitiesChanged = true;
    
            // TODO Push resumeArgs into the activity for consideration
            r = performResumeActivity(token, clearHide, reason);
    
            if (r != null) {
                final Activity a = r.activity;
    
                if (localLOGV) Slog.v(
                    TAG, "Resume " + r + " started activity: " +
                    a.mStartedActivity + ", hideForNow: " + r.hideForNow
                    + ", finished: " + a.mFinished);
    
                final int forwardBit = isForward ?
                        WindowManager.LayoutParams.SOFT_INPUT_IS_FORWARD_NAVIGATION : 0;
    
                // If the window hasn't yet been added to the window manager,
                // and this guy didn't finish itself or start another activity,
                // then go ahead and add the window.
                boolean willBeVisible = !a.mStartedActivity;
                if (!willBeVisible) {
                    try {
                        willBeVisible = ActivityManager.getService().willActivityBeVisible(
                                a.getActivityToken());
                    } catch (RemoteException e) {
                        throw e.rethrowFromSystemServer();
                    }
                }
                ...
                    r.activity.mVisibleFromServer = true;
                    mNumVisibleActivities++;
                    if (r.activity.mVisibleFromClient) {
                        r.activity.makeVisible();
                    }
                }
                ...
            }
        }
    

    可以看到里面调用了performResumeActivity()方法,继续跟到performResumeActivity()方法体:

     public final ActivityClientRecord performResumeActivity(IBinder token,
                boolean clearHide, String reason) {
            ActivityClientRecord r = mActivities.get(token);
            if (localLOGV) Slog.v(TAG, "Performing resume of " + r
                    + " finished=" + r.activity.mFinished);
            ...
                    r.activity.performResume();
    
                    synchronized (mResourcesManager) {
                        // If there is a pending local relaunch that was requested when the activity was
                        // paused, it will put the activity into paused state when it finally happens.
                        // Since the activity resumed before being relaunched, we don't want that to
                        // happen, so we need to clear the request to relaunch paused.
                        for (int i = mRelaunchingActivities.size() - 1; i >= 0; i--) {
                            final ActivityClientRecord relaunching = mRelaunchingActivities.get(i);
                            if (relaunching.token == r.token
                                    && relaunching.onlyLocalRequest && relaunching.startsNotResumed) {
                                relaunching.startsNotResumed = false;
                            }
                        }
                    }
    
                   ...
                }
            }
            return r;
        }
    

    performResumeActivity()方法调用了r.activity.performResume(),我们继续看Activity的performResume()的源码,再次深入,再次快乐。

    final void performResume() {
           ...
    
            mCalled = false;
            // mResumed is set by the instrumentation
            mInstrumentation.callActivityOnResume(this);
            if (!mCalled) {
                throw new SuperNotCalledException(
                    "Activity " + mComponent.toShortString() +
                    " did not call through to super.onResume()");
            }
    
           ...
        }
    

    然后又调用了Instrumentation的callActivityOnResume方法,继续看该方法的源码,一次到底,持续快乐:

     public void callActivityOnResume(Activity activity) {
            activity.mResumed = true;
            activity.onResume();
    
            if (mActivityMonitors != null) {
                synchronized (mSync) {
                    final int N = mActivityMonitors.size();
                    for (int i=0; i<N; i++) {
                        final ActivityMonitor am = mActivityMonitors.get(i);
                        am.match(activity, activity, activity.getIntent());
                    }
                }
            }
        }
    

    可以看到callActivityOnResume()方法调用了activity.onResume(),即回调到Activity的onResume()方法,综合上面的分析可以得出:ViewRootImpl是在Activity的OnResume()方法后面创建出来的。

    到这里可以事后一支烟了,不是,是总结一下了:
    1)ViewRootImpl是在Activity的onResume()方法后面创建出来的,所以在onResume之前的UI更新可以在子线程操作而不报错,因为这个时候ViewRootImpl还没有创建,没有执行checkThread()方法。
    2)安卓系统中,操作viwe对象没有加锁,所以如果在子线程中更新UI,会出现多线程并发的问题,导致页面展示异常。

    小伙子分析得很不错,把我打动了,回去等offer吧。

    作者:天才少年_
    链接:https://juejin.im/post/5ec08467f265da7bda415343

    相关文章

      网友评论

        本文标题:面试官问:Android中子线程为什么不能更新UI,该怎么答?

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