一:报错情况
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结束之后,又把进入标志改回来。
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,在子线程中也可以修改文字不报错!!试一试!!