1.
测试类:
public class AndroidThreadDemo extends Activity {
TextView text;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.android_thread);
text = (TextView) findViewById(R.id.text);
thread.start();
}
Thread thread = new Thread(){
public void run() {
// try {
// Thread.sleep(500);
// } catch (InterruptedException e) {
// e.printStackTrace();
// }
text.setText("success");
};
};
}
布局文件中只有一个TextView,编译运行后thread可以更新text的内容。但是当启用Thread.sleep(500)这行代码,却抛出异常:
E/AndroidRuntime(1337): android.view.ViewRootImpl$CalledFromWrongThreadException: Only the original thread that created a view hierarchy can touch its views.
原因在于:ActivityThread在Activity的onResume()中才会为Activity分配ViewRootImpl对象,同时指定主线程(UI线程)。而View的invalidate()方法是在UI线程中执行的。所以在onCreate中可以直接通过线程更新UI,前提是线程执行的时间足够短,要在onResume()回调之前执行完。
当启用Thread.sleep(500)后,由于线程休眠,onResume()会在thread执行完之前回调,在onResume()已完成主线程的分配,此时在非UI线程中更新UI就会抛出异常。
具体可参考ActivityThread, ViewRootImpl, View的invalidate()方法。
2.
Android为什么只能在UI线程中更新UI?
首先假设可以在非UI线程中更新UI,那么多个线程同时去更新UI,就会导致线程并发问题。Android只允许在UI线程中更新UI,使UI更新更简单,开发更容易。