Android UI更新只能在主线程工作是个美丽的谎言?

最近看到某博客出现了一个关于非主线程更新view的一个案例,这个有兴趣的朋友可以搜索一下,这里借鉴使用一下demo

 protected void onCreate(Bundle savedInstanceState) {
        super.onCreate(savedInstanceState);
        setContentView(R.layout.activity_main);
        textView= (TextView) findViewById(R.id.tv);
         new Thread(new Runnable() {
        @Override
        public void run() {
            textView.setText("子线程设置");
        }
    }).start();
    }

运行这个代码,竟然神奇的没有出现报错!
然而,当你替换代码

new Thread(new Runnable() {
        @Override
        public void run() {
            try{
                Thread.sleep(100);
            }catch (Exception e){
                e.printStackTrace();
            }
            textView.setText("子线程设置");
        }
    }).start();

出现如下错误:

FATAL EXCEPTION: Thread-1055
                                                 Process: com.tianyuan.viewthread, PID: 7532
                                                 android.view.ViewRootImpl$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.
                                                     at android.view.ViewRootImpl.checkThread(ViewRootImpl.java:6495)
                                                     at android.view.ViewRootImpl.requestLayout(ViewRootImpl.java:904)
                                                     at android.view.View.requestLayout(View.java:17534)
                                                     at android.view.View.requestLayout(View.java:17534)
                                                     at android.view.View.requestLayout(View.java:17534)
                                                     at android.view.View.requestLayout(View.java:17534)
                                                     at android.view.View.requestLayout(View.java:17534)
                                                     at android.view.View.requestLayout(View.java:17534)
                                                     at android.view.View.requestLayout(View.java:17534)
                                                     at android.widget.TextView.checkForRelayout(TextView.java:6952)
                                                     at android.widget.TextView.setText(TextView.java:4080)
                                                     at android.widget.TextView.setText(TextView.java:3938)
                                                     at android.widget.TextView.setText(TextView.java:3913)
                                                     at com.tianyuan.viewthread.MainActivity$1.run(MainActivity.java:25)

为什么前后会出现如此的区别呢?那么我们根据报错日志看看,报错位置是在android.view.ViewRootImpl 中间有个check Thread方法,代码如下
这里写图片描述

我们仔细查看viewrootimpl方法后发现,这个方法会在一些重要位置检查Thread,viewrootimpl对于view tree进行perfrom traversals ,view tree的初始化是从viewrootimpl执行perform traversals开始的,这个过程遍历view tree节点。
通过debug,我们可以看到view在执行一些操作大体步骤为:
view.invalidate
view.invalidateinternal
Viewgroup.invalidatechild
viewgroup.invalidateChildINparent
viewrootimpl.invalidateChildinparent
viewrootimpl.checkThread

当activity初始化时,viewroot没有完整的viewtree,view的mAttachinfo还未初始化完成。我们看到很多方法是这么的

   if (mAttachInfo == null) {
                return;       }
 /**
     * A set of information given to a view when it is attached to its parent
     * window.
     */
    final static class AttachInfo {
        interface Callbacks {
            void playSoundEffect(int effectId);
            boolean performHapticFeedback(int effectId, boolean always);
        }

由此我们知道了,android更新UI之所以不能再主线程更新,是因为view这个类中viewrootimpl对于更新关键的操作进行了线程判断而导致的,在某些初始化未完成的时候,也是可以执行的,不过是特例而已。

你可能感兴趣的:(Android基础)