Android 中非UI线程真的不能更新UI吗?

来源:互联网 发布:数据采集器厂家 编辑:程序博客网 时间:2024/05/31 18:31

参考文献:

http://android.jobbole.com/85148/

先上一段代码:

public class MainActivity extends AppCompatActivity {    private TextView main_tv;    @Override    protected void onCreate(Bundle savedInstanceState) {        super.onCreate(savedInstanceState);        setContentView(R.layout.activity_main);        main_tv = (TextView) findViewById(R.id.main_tv);        new Thread(new Runnable() {            @Override            public void run() {                try {                //关键在这里                //如果加上这一句话,程序运行崩溃                //如果不加这一句话,居然就会在子线程里面更新UI                    Thread.sleep(200);                } catch (InterruptedException e) {                    e.printStackTrace();                }                main_tv.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:6581)
at android.view.ViewRootImpl.requestLayout(ViewRootImpl.java:924)

……

那么问题出现了:同样的代码,为什么一个崩溃,一个就能更新UI?

本着拿来主义的原则,这里直接上结论了。

看一下异常信息:

at android.view.ViewRootImpl.checkThread(ViewRootImpl.java:6581)

这是因为ViewRootImpl的checkThread()方法会检查更新UI的线程是不是UI线程。如果不是,就会抛出异常。

经过研究源码发现,ViewRootImpl是在onResume方法回调之后创建的。所以,对于没有崩溃的原因就是,子线程在onCreate中更新UI的时候,ViewRootImpl还没有创建。反过来,崩溃的原因就是子线程在Thread.sleep(200)后,ViewRootImpl已经创建了,并且通过checkThread()抛出的异常。

附:更新UI需要UI线程,主线程就是一个UI线程。因为还可以创建很多UI线程,所以不只有主线程才能更新UI。
但是一般的子线程是不能更新UI的。本文中提到的“一般的子线程”处于一个特殊的位置,所以才更新了UI。

0 0
原创粉丝点击