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-11-07 16:31:02

    nginx配置X-Frame-Options允许多个域名iframe嵌套

    有时候我们需要允许多个url的来源,但是又不能全部开放,所以应该匹配第三种方法ALLOW-FROM url,那么多个url该如何配置呢,百度了所有网站都没有找到,那么这里写给大家,其实很简单: add_header X-Frame-Options 'ALLOW-FROM https://xxx.xxxxxx.com https://xxx2.xxxxxxx.com'; 就是使用空格隔开就好了!

  • 2020-11-08 08:31:51

    meteor在不同端口启动服务

    当没有任何参数时,run是默认行为,在幕后,它3000端口开启node.js服务器实例,同时开启监听3001端口的MongoDB服务

  • 2020-11-11 15:05:39

    nuxt如何在其它js文件中使用store

    在新建的js文件中想用store里面的数据,比如token想在封装的axios里面,请求头里面去使用,亦或者通过app的JS接口获取token并存储在store里面。我们都知道如何在vue中如何使用。

  • 2020-11-12 14:01:46

    使用postMessage来实现父子通信跨域

    1.子向父,子postMessage,父监听message; 2.父向子,父postMessage,子监听message; 3.测试发现,子向父postMessage的时候,源可以写为‘*’,父向子postMessage的时候,源需要写成子的源,(也就是子页面的协议+主机号+端口) 测试代码部分:

  • 2020-11-12 14:24:39

    Object.entries()

    Object.entries()方法返回一个给定对象自身可枚举属性的键值对数组,其排列与使用 for...in 循环遍历该对象时返回的顺序一致(区别在于 for-in 循环还会枚举原型链中的属性)