Android 更新UI 只能在主线程?
2016-09-13 20:22
309 查看
今天中午去吃饭的时候,zk问了我一个问题,“Android只能在UI线程更新UI 么”,我的回答是“对”。然后zk让我回去写在子线程中更新UI,看会有什么问题。
主UI中含有三个Button,其点击事件分别对应着三个不同的子线程。
btNew点击事件,报错信息是:
btNew2点击事件,报错信息是:
btNew3点击事件,正常展示,在子线程中更新了UI。
线程能否刷新UI的关键在于ViewRoot是否属于该线程。
首先,CalledFromWrongThreadException这个异常是由下面的代码抛出的:
依次调用顺序是:
因此知道,调用btNew.setText()报CalledFromWrongThreadException(这个请求来自于错误的线程)的原因是,调用该方法所在的线程与btNew对应的ViewRoot所初始化的线程不是同一个线程。能否更新UI与是否是工作线程、主线程没有关系。取决于该线程拥有自己的ViewRoot。
而btNew3事件中,通过 Looper.prepare(),在自己的线程中创建了WindowManager并与当前线程对应,因此可以更新UI。
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 )
一、三个子线程更新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、使用Handler2、用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 )
相关文章推荐
- Android中关于“UI只能在主线程中更新”说法的理解
- Android中在主线程与在子线程中更新UI的探索
- “只能在UI主线程更新View”这件小事,android ui
- android 只有主线程能更新UI
- Android中通过其他线程更新主线程UI
- Android只有主线程才能更新UI?
- 【Android】快速切换到主线程更新UI的几种方法
- Android Activity sleep 自线程更新主线程UI
- android线程间通信和主线程更新ui
- Android中为什么主线程更新UI,子线程执行耗时操作?
- Android中通过其他线程更新主线程UI
- Android 在子线程中更新主线程UI
- 【Android】快速切换到主线程更新UI的几种方法
- 【Android和iOS】快速切换到主线程更新UI
- 主线程负责交互(事件处理)和UI绘制(onDraw), 只能在主线程访问View, 其他线程不能直接访问View (Android)
- Android中通过其他线程更新主线程UI
- Android线程模型解析(包括UI的更新)
- Android线程模型解析(包括UI的更新)
- android异步更新UI
- Android中实现view的更新UI有两组方法