您的位置:首页 > 产品设计 > UI/UE

Android 更新UI 只能在主线程?

2016-09-13 20:22 309 查看
今天中午去吃饭的时候,zk问了我一个问题,“Android只能在UI线程更新UI 么”,我的回答是“对”。然后zk让我回去写在子线程中更新UI,看会有什么问题。

一、三个子线程更新UI

下午空闲的时候,就带着zk的疑问,写了这个DEMO,代码如下:

package com.troy.nouithread;

import android.graphics.PixelFormat;
import android.os.Bundle;
import android.os.Looper;
import android.support.v7.app.AppCompatActivity;
import android.view.Gravity;
import android.view.View;
import android.view.WindowManager;
import android.widget.Button;
import android.widget.TextView;

public class MainActivity extends AppCompatActivity {

private Button btNew;
private Button btNew2;
private Button btNew3;
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
setContentView(R.layout.activity_main);
btNew = (Button)findViewById(R.id.btNew);
btNew2 = (Button)findViewById(R.id.btNew2);
btNew3 = (Button)findViewById(R.id.btNew3);
btNew.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
newThread();
}
});

btNew2.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
newThread2();
}
});

btNew3.setOnClickListener(new View.OnClickListener() {
@Override
public void onClick(View v) {
new TestThread().start();
}
});
}

private void newThread(){
new Thread(new Runnable() {
@Override
public void run() {
WindowManager windowManager = getWindowManager();
TextView textView = new TextView(getApplicationContext());
textView.setText("在工作线程更新UI");
textView.setTextColor(getResources().getColor(R.color.colorPrimary));
windowManager.addView(textView, new WindowManager.LayoutParams());
}
}).start();
}

private void newThread2(){
new Thread(new Runnable() {
@Override
public void run() {
// btNew.setBackgroundColor(getResources().getColor(R.color.colorPrimary));
btNew.setText("选择比努力重要");
}
}).start();
}

class TestThread extends Thread{
@Override
public void run() {
Looper.prepare();
TextView tx = new TextView(MainActivity.this);
tx.setText("时间煮雨");
tx.setTextColor(getResources().getColor(R.color.white));
tx.setBackgroundColor(getResources().getColor(R.color.colorPrimary));
tx.setGravity(Gravity.CENTER);
WindowManager wm = MainActivity.this.getWindowManager();
WindowManager.LayoutParams params = new WindowManager.LayoutParams(
250, 150, 200, 200, WindowManager.LayoutParams.FIRST_SUB_WINDOW,
WindowManager.LayoutParams.TYPE_TOAST, PixelFormat.OPAQUE);
wm.addView(tx, params);
Looper.loop();
}
}
}


主UI中含有三个Button,其点击事件分别对应着三个不同的子线程。

btNew点击事件,报错信息是:



btNew2点击事件,报错信息是:



btNew3点击事件,正常展示,在子线程中更新了UI。

二、Android UI 线程检查机制

分析CalledFromWrongThreadException

在事件2中,出现的错误信息是:

CalledFromWrongThreadException:Only the original thread that created a view hierarchy can touch its views.


线程能否刷新UI的关键在于ViewRoot是否属于该线程。

首先,CalledFromWrongThreadException这个异常是由下面的代码抛出的:

public final class ViewRootImpl implements ViewParent{
...
void checkThread() {
if (mThread != Thread.currentThread()) {
throw new CalledFromWrongThreadException(//这个请求来自于错误的线程
"Only the original thread that created a view hierarchy can touch its views.");
//只有最初创建视图层次结构的线程才可以接触到这些视图。
}
}
...
}


依次调用顺序是:

-> android.widget.TextView.setText

-> android.widget.TextView.checkForRelayout

-> android.view.View.invalidate

-> android.view.ViewGroup.invalidateChild

-> android.view.ViewRootImpl.invalidateChildInParent

-> android.view.ViewRootImpl.invalidateChild

-> android.view.ViewRootImpl.checkThread


因此知道,调用btNew.setText()报CalledFromWrongThreadException(这个请求来自于错误的线程)的原因是,调用该方法所在的线程与btNew对应的ViewRoot所初始化的线程不是同一个线程。能否更新UI与是否是工作线程、主线程没有关系。取决于该线程拥有自己的ViewRoot。

而btNew3事件中,通过 Looper.prepare(),在自己的线程中创建了WindowManager并与当前线程对应,因此可以更新UI。

三、在子线程中更新UI的五个策略

1、使用Handler

2、用Activity对象的runOnUiThread方法

3、View.post(Runnable r)

4、Broadcast子线程中发送广播,主线程中接收广播并更新UI

5、AsyncTask

参考致谢:

(1)、Android里子线程真的不能刷新UI吗?

(2)、为什么我们可以在非UI线程中更新UI

(3)、Android子线程中更新UI的几种方法

(4)、Android View.post(Runnable )
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: