如何在子线程中更新UI

一:报错情况

 android.view.ViewRootImpl$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.
        at android.view.ViewRootImpl.checkThread(ViewRootImpl.java:8798)
        at android.view.ViewRootImpl.requestLayout(ViewRootImpl.java:1606)
        at android.view.View.requestLayout(View.java:25390)
        at android.view.View.requestLayout(View.java:25390)
        at android.view.View.requestLayout(View.java:25390)
        at android.view.View.requestLayout(View.java:25390)
        at android.view.View.requestLayout(View.java:25390)
        at android.view.View.requestLayout(View.java:25390)
        at androidx.constraintlayout.widget.ConstraintLayout.requestLayout(ConstraintLayout.java:3593)
        at android.view.View.requestLayout(View.java:25390)
        at android.widget.TextView.checkForRelayout(TextView.java:9719)
        at android.widget.TextView.setText(TextView.java:6311)

我尝试在子线程中更新UI:

binding.textView.setOnClickListener {
            thread {
                (it as TextView).text = "ldkjfla;66666sdf"
            }
        }

二:报错原因

首先,我们更新UI,会调用text view的request layout方法, 然后view 的request layout方法又会调用到它父view的 request layout方法:

子view request layout  ------>    父view request layout

这样一层层调用上去,因为view系统的最上层是一个叫作view root impl的view,所以最终会调用到它的request layout方法。

我们来看看它的request layout方法,然后看看有没有什么对策:

@Override
    public void requestLayout() {
        if (!mHandlingLayoutInLayoutRequest) {
            checkThread();  和之前报错代码的最上面对应起来了
            mLayoutRequested = true;
            scheduleTraversals();
        }
    }

注意啦注意啦!!当在子view中更新UI,就会调用到view root impl的request layout方法!!然后里面会调用check thread方法来看看更新UI的线程等不等于view root impl创建时的线程!!

view root impl是在activity的onResume生命周期主线程中创建的。所以这个checkThread就不通过啦!!

三:解决办法

首先,上面说,更新UI时,会调用request layout方法一层层调用上去,那我们去看看这个路径,看看有没有办法斩断这个路径。

 1:进入text view的set text方法

2:进入check for relayout方法

3:进入 request layout方法

4:然后就进入了view的request layout方法之中

然后接下来的过程就是我刚刚说的,一直向上调用到view root impl的request layout最后报错了。注意啦!!注意啦!!在判断向不向上去传递调用request layout的时候,会看看

!mParent.isLayoutRequested()

如果我们让view root impl的LayourRequested参数为true,然后表达式为false,就不会调用到view root impl的requst layout方法,就不会check thread了。

所以我们要让这个参数为true!!!

四:让这个参数为true

我们可以注意到, view root impl的request layout方法中,在check thread之后,顺手就把这个参数置为true了。所以我们可以调用一次textview的request layout方法,然后调用到view root impl的方法,把它置为true,然后我们再在子线程更新UI,就不会进入view root impl的request layout方法了。

binding.textView.setOnClickListener {
            it.requestLayout()//因为在主线程,所以最后check thread没有事
            thread {
                (it as TextView).text = "ldkjfla;66666sdf"
            }
        }

五:LayoutRequested的意义

 我们更新UI,就会调用到view root impl的request layout方法,在check thread之后,就会把Layoutrequested置为true!!表示自己正处于被请求重新去布局的状态!!置为true,之后,下一个方法就是鼎鼎有名的

scheduleTraversals()

它会执行view root impl的performTraversal!!  对整个view tree进行从上到下的测量、布局和绘制!!在这个perform traversal结束时,会把LayoutRequested置为false。不然,下一次更新UI不就会没办法到这个方法然后失败了嘛!!所以为了性能原因,在打算开展一个perform traversal之前,会把进入标志改一下,perform traversal结束之后,又把进入标志改回来。

我们就是趁它标志位为true的时候更改UI.
要是它perform traversal结束之后把标志位改了回来,,,那还是会报错。
binding.textView.setOnClickListener {
            it.requestLayout()//因为在主线程,所以最后check thread没有事
            thread {
                sleep(1000)//等了一秒, performTraversal结束之后标志位恢复,又会去到check thread了。
                (it as TextView).text = "ldkjfla;66666sdf"
            }
        }

番外:

对于text view,在checkForRelayout方法中,会看看宽高是否改变然后决定是否向上传递request layout,所以,如果一个text view固定宽高,即使不主动request layout,在子线程中也可以修改文字不报错!!试一试!!

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值