如果之前你没有尝试过onCreate方法里面用子线程的run方法去设置UI(比如对Textview进行setText操作)
在相信你看到这个标题,也会感到困惑和好奇吧。
废话不多说,先来个Demo。
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
tv = (TextView) findViewById(R.id.tv);
iv = (ImageView) findViewById(R.id.iv);
bt = (Button) findViewById(R.id.bt);
Thread thread = new Thread("Thread1") {
public void run() {
tv.setText(System.currentTimeMillis() + "33333333");
tv.append(Thread.currentThread().getName());
iv.setImageResource(R.drawable.ic_launcher);
};
};
thread.start();
}
代码很简单,就是在onCreate方法里面new了一个Thread,然后再run方法里面进行UI操作。
运行结果如下:
没错,你要相信这正是运行结果。确实TextView的文本设置为当前时间的long型,还有子线程的名字。ImageView也设置了默认图标
那,不是说android不允许子线程更新UI吗?
在这里,我们不妨再加一个Button,在Button点击事件里面通过子线程来更新UI
onClick方法代码如下:
public void onClick(View v) {
// TODO 自动生成的方法存根
Thread thread = new Thread("Thread1") {
public void run() {
tv.setText(System.currentTimeMillis() + "\n");
tv.append("thread name:"
+ Thread.currentThread().getName());
iv.setImageResource(R.drawable.ic_launcher);
};
};
thread.start();
}
编译后运行结果来看:
编译后程序可以启动,并且没有挂掉,界面上TextView和ImageView显示与上图一致。
但是我们点击按钮后发现程序出现“ Unforunately,Study has stopped.”的字样,程序挂掉了。
不难看出,onClick里面子线程更新UI与我们的预期结果一样,但是为什么onCreate方法里面却没有挂掉呢。
在此,我选择查看一下setText方法,发现setText里面调用了 checkForRelayout();,而在checkForRelayout方法里面调用了invalidate()这个方法
再追踪下去,是invalidate方法调用了ViewGroup.invalidateChild,最终调用ViewRootImpl.checkThread()。
ViewRootImpl是一个隐藏类,我们只能去看framework的源码,源码如下:
void checkThread() {
if (mThread != Thread.currentThread()) {
throw new CalledFromWrongThreadException(
"Only the original thread that created a view hierarchy can touch its views.");
}
}
mThread是UI线程,这里会检查当前线程是不是UI线程。那么为什么onCreate里面没有进行这个检查呢。
这个问题原因出现在Activity的生命周期中,在onCreate方法中,UI处于创建过程,对用户来说界面还不可视,直到onStart方法后界面可视了,再到onResume方法后界面可以交互,。
从某种程度来讲,在onCreate方法中进行setText不能算是更新UI,只能说是配置UI,或者是设置UI的属性。这个时候并不会调用invalidate,也就是不会调用到ViewRootImpl.checkThread()。而在onResume方法后,ViewRootImpl才被创建。这个时候去交互界面以及算是更新UI了,这个时候ViewRootImpl.checkThread()就会报错了。
不信,我们把子线程休眠200ms,代码如下:
Thread thread = new Thread("Thread1") {
public void run() {
try {
Thread.sleep(200);
} catch (InterruptedException e) {
// TODO 自动生成的 catch 块
e.printStackTrace();
}
tv.append( System.currentTimeMillis() + "\n");
tv.append("thread name:" + Thread.currentThread().getName());
iv.setImageResource(R.drawable.ic_launcher);
};
};
thread.start();
运行结果是程序出现“Unforunately,Study has stopped
简单的来说,就是多线程下的问题,在activity的生命周期里各个方法的执行时间都有影响,而UI线程检查是由ViewRootImpl类调用的,而ViewRootImpl只有在onRsume方法后才会被创建。