android中Activity的onCreate方法里面子线程为何能设置UI界面

如果之前你没有尝试过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方法后才会被创建。

评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值