Android子线程中更新UI的3种方法

2018-11-18 09:06:06

在Android项目中经常有碰到这样的问题,在子线程中完成耗时操作之后要更新UI,下面就自己经历的一些项目总结一下更新的方法。

一. 引言

首先来看一下Android中消息机制

专业术语:

Message:消息,其中包含了消息ID,消息处理对象以及处理的数据等,由MessageQueue统一列队,终由Handler处理。 
Handler:处理者,负责Message的发送及处理。使用Handler时,需要实现handleMessage(Message msg)方法来对特定的Message进行处理,例如更新UI等。 
MessageQueue:消息队列,用来存放Handler发送过来的消息,并按照FIFO规则执行。当然,存放Message并非实际意义的保存,而是将Message以链表的方式串联起来的,等待Looper的抽取。 
Looper:消息泵,不断地从MessageQueue中抽取Message执行。因此,一个MessageQueue需要一个Looper。 
Thread:线程,负责调度整个消息循环,即消息循环的执行场所。 

二. 方法   

 1. 用Handler

(1) 主线程中定义Handler: 

java代码:

Handler mHandler = new Handler() {@Override  public void handleMessage(Message msg) {super.handleMessage(msg);switch (msg.what) {case 0://完成主界面更新,拿到数据  String data = (String)msg.obj;updateWeather();textView.setText(data);break;default:break;}}};


(2)子线程发消息,通知Handler完成UI更新: 

java代码:

private void updateWeather() {new Thread(new Runnable(){@Override  public void run() {//耗时操作,完成之后发送消息给Handler,完成UI更新;  mHandler.sendEmptyMessage(0);//需要数据传递,用下面方法;  Message msg =new Message();msg.obj = "数据";//可以是基本类型,可以是对象,可以是List、map等;  mHandler.sendMessage(msg);}}).start();}


注意:Handler对象必须定义在主线程中,如果是多个类直接互相调用,就不是很方便,需要传递content对象或通过接口调用。

2. 用Activity对象的runOnUiThread方法更新 

在子线程中通过runOnUiThread()方法更新UI: 

java代码:

new Thread() {public void run() {//这儿是耗时操作,完成之后更新UI;  runOnUiThread(new Runnable(){@Override  public void run() {//更新UI  imageView.setImageBitmap(bitmap);}});}}.start();

如果在非上下文类中,可以通过传递上下文实现调用:

java代码:

Activity activity = (Activity) imageView.getContext();activity.runOnUiThread(new Runnable() {@Override  public void run() {imageView.setImageBitmap(bitmap);}});
注意:这种方法使用比较灵活,但如果Thread定义在其他地方,需要传递Activity对象。

3. View.post(Runnable r) 

java代码:

imageView.post(new Runnable(){@Override  public void run() {imageView.setImageBitmap(bitmap);}});
这种方法更简单,但需要传递要更新的View过去。

总结:UI的更新必须在主线程中完成,所以不管上述那种方法,都是将更新UI的消息发送到了主线程的消息对象,让主线程做处理。


  • 2020-01-14 00:12:22

    webpack externals详解

    在众多的webpack配置教程中,对externals这个配置选项,总是一带而过,把文档中提到的几种方式都复述一遍,但是对于开发者而言,根本没法完全理解。本文试图通过一整篇文章,详细的对externals这个参数进行讲解。

  • 2020-01-14 01:06:37

    webpack externals 深入理解

    按照官方文档的解释,如果我们想引用一个库,但是又不想让webpack打包,并且又不影响我们在程序中以CMD、AMD或者window/global全局等方式进行使用,那就可以通过配置externals。这个功能主要是用在创建一个库的时候用的,但是也可以在我们项目开发中充分使用。

  • 2020-01-14 01:08:19

    webpack用externals优化echarts

    防止将某些 import 的包(package)打包到 bundle 中,而是在运行时(runtime)再去从外部获取这些扩展依赖(external dependencies)。

  • 2020-01-16 08:52:22

    Vue函数式调用组件创建公共组件

    所有组件都需要这么去调用,就会有些许麻烦而且不太美观。像Loading、Toast等这些组件,一页面可以经常用到而且每次显示的内容都可能不一样,这样的话用js的方式【this.$xxx.show(option)】去调用就方便很多,而且代码也更整洁。

  • 2020-01-17 08:37:26

    css transition分别指定多个属性

    transition有四个属性,很多人都会遗忘,分别是transition-property,transition-duration,transition-timing-function,transition-delay,尤其是transition-delay,这个可以实现延迟动画