在平时的Android开发中,如果一个新手遇到一个这样的错:
android.view.ViewRootImpl$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.
at android.view.ViewRootImpl.checkThread(ViewRootImpl.java:8066)
at android.view.ViewRootImpl.requestLayout(ViewRootImpl.java:1297)
at android.view.View.requestLayout(View.java:23147)
你作为一只老鸟,嘴角露出一丝微笑:
“小兄弟,你这个是没有在UI线程执行UI操作导致的错误,你搞个UI线程的handler.post一下就好了”。
但是...
我今天要说,真是是只有UI线程才能更新UI吗?
你作为一只老鸟,肯定立马脑子里闪过:
我知道你这文章写啥了,又要在Activity#onCreate,去搞个线程执行TextView#setText,然后发现更新成功了,是不是?
这多年以前我就看过这样的文章,ViewRootImpl还没创建而已。
看你们这么强,我这个文章没法写下去了...
但是我这个人专治各种不服好吧,我换个问题:
UI线程更新UI就不会出现上面的错误了吗?
好了,开讲。
下面是一个应届小哥小奇写需求的故事。
注意本文代码为应届小哥角度所写,为了引出问题及原理,不要随意参考,另外如果尝试复现相关代码,务必看好每一个字符,甚至xml里面的属性都很关键。
小哥的需求
需求很简单,就是
- 点击一个按钮;
- Server会下发一个问题,客户端Dialog展示;
- 在Dialog交互回答问题;
是不是很简答。
小哥怒写一波代码:
package com.example.testviewrootimpl;
import androidx.appcompat.app.AppCompatActivity;
import android.os.Bundle;
import android.view.View;
import android.widget.Button;
public class MainActivity extends AppCompatActivity {
private Button mBtnQuestion;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
mBtnQuestion = findViewById(R.id.btn_question);
mBtnQuestion.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View view) {
requestAQuestion();
}
});
}
private void requestAQuestion() {
new Thread(){
@Override
public void run() {
try {
Thread.sleep(1000);
} catch (InterruptedException e) {
e.printStackTrace();
}
// 模拟服务器请求,返回问题
String title = "鸿洋帅气吗?";
showQuestionInDialog(title);
}
}.start();
}
private void showQuestionInDialog(String title) {
}
}
很简单吧,点击按钮,新启动一个线程去模拟网络请求,结果拿到后,把问题展示在Dialog。
下面开始写Dialog的代码:
public class QuestionDialog extends Dialog {
private TextView mTvTitle;
private Button mBtnYes;
private Button mBtnNo;
public QuestionDialog(@NonNull Context context) {
super(context);
setContentView(R.layout.dialog_question);
mTvTitle = findViewById(R.id.tv_title);
mBtnYes = findViewById(R.id.btn_yes);
mBtnNo = findViewById(R.id.btn_no);
}
public void show(String title) {
mTvTitle.setText(title);
show();
}
}
很简答,就一个标题,两个按钮。
<RelativeLayout xmlns:android="http://schemas.android.com/apk/res/android"
xmlns:tools="http://schemas.android.com/tools"
android:layout_width="match_parent"
android:layout_height="match_parent">
<TextView
android:id="@+id/tv_title"
android:layout_width="match_parent"
android:layout_height="