今天中午去吃饭的时候,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点击事件,报错信息是:

Android 更新UI 只能在主线程?_第1张图片

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

Android 更新UI 只能在主线程?_第2张图片

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 )

更多相关文章

  1. Android onTouch事件解析
  2. Android 并发工具类与线程池
  3. android中的进程与线程
  4. Android线程处理简述
  5. Android 基于回调的事件处理----事件分发

随机推荐

  1. TopGeek:移动互联网时代,选择Android还是iO
  2. Android 获取最近应用的缩略图
  3. Motorola RAZR 將會在 2012 年初更新到 I
  4. 国内Android Market孰更好
  5. Android编程之客户端通过socket与服务器
  6. 在android上运行C程序
  7. 了解Android 的支持库
  8. Android之monkey Test,Monkey测试中的黑
  9. Android邮件发送详解
  10. Android开发实践:自己动手编写图片剪裁应