Android之Message,MessageQueue,Looper,Handler详解(带示例)

2024-01-06 03:38

本文主要是介绍Android之Message,MessageQueue,Looper,Handler详解(带示例),希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

一、几个关键概念


1、MessageQueue:是一种数据结构,见名知义,就是一个消息队列,存放消息的地方。每一个线程最多只可以拥有一个MessageQueue数据结构。
创建一个线程的时候,并不会自动创建其MessageQueue。通常使用一个Looper对象对该线程的MessageQueue进行管理。主线程创建时,会创建一
个默认的Looper对象,而Looper对象的创建,将自动创建一个Message Queue。其他非主线程,不会自动创建Looper,要需要的时候,通过调
用prepare函数来实现。

2、Message:消息对象,Message Queue中的存放的对象。一个Message Queue中包含多个Message。
Message实例对象的取得,通常使用Message类里的静态方法obtain(),该方法有多个重载版本可供选择;它的创建并不一定是直接创建一个新的实例,
而是先从Message Pool(消息池)中看有没有可用的Message实例,存在则直接取出返回这个实例。如果Message Pool中没有可用的Message实例,
则才用给定的参数创建一个Message对象。调用removeMessages()时,将Message从Message Queue中删除,同时放入到Message Pool中。除了上面这
种方式,也可以通过Handler对象的obtainMessage()获取一个Message实例。

3、Looper:
是MessageQueue的管理者。每一个MessageQueue都不能脱离Looper而存在,Looper对象的创建是通过prepare函数来实现的。同时每一个Looper对象
和一个线程关联。通过调用Looper.myLooper()可以获得当前线程的Looper对象
创建一个Looper对象时,会同时创建一个MessageQueue对象。除了主线程有默认的Looper,其他线程默认是没有MessageQueue对象的,所以,不能
接受Message。如需要接受,自己定义一个Looper对象(通过prepare函数),这样该线程就有了自己的Looper对象和MessageQueue数据结构了。
Looper从MessageQueue中取出Message然后,交由Handler的handleMessage进行处理。处理完成后,调用Message.recycle()将其放入Message Pool中。

4、Handler:
消息的处理者,handler负责将需要传递的信息封装成Message,通过调用handler对象的obtainMessage()来实现;
将消息传递给Looper,这是通过handler对象的sendMessage()来实现的。继而由Looper将Message放入MessageQueue中。
当Looper对象看到MessageQueue中含有Message,就将其广播出去。该handler对象收到该消息后,调用相应的handler对象的handleMessage()方法
对其进行处理。
二、线程之间的消息如何进行传递

1、主线程给自己发送Message


package test.message;import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.Looper;
import android.os.Message;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;public class MainActivity extends Activity {private Button btnTest;private TextView textView;private Handler handler;@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.main);btnTest = (Button)this.findViewById(R.id.btn_01);textView = (TextView)this.findViewById(R.id.view_01);btnTest.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View arg0) {Looper looper = Looper.getMainLooper(); //主线程的Looper对象//这里以主线程的Looper对象创建了handler,//所以,这个handler发送的Message会被传递给主线程的MessageQueue。handler = new MyHandler(looper);handler.removeMessages(0);//构建Message对象//第一个参数:是自己指定的message代号,方便在handler选择性地接收//第二三个参数没有什么意义//第四个参数需要封装的对象Message msg = handler.obtainMessage(1,1,1,"主线程发消息了");handler.sendMessage(msg); //发送消息}});}class MyHandler extends Handler{public MyHandler(Looper looper){super(looper);}public void handleMessage(Message msg){super.handleMessage(msg);textView.setText("我是主线程的Handler,收到了消息:"+(String)msg.obj);}}
}

2、其他线程给主线程发送Message


package test.message;import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.Looper;
import android.os.Message;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;public class MainActivity extends Activity {private Button btnTest;private TextView textView;private Handler handler;@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.main);btnTest = (Button)this.findViewById(R.id.btn_01);textView = (TextView)this.findViewById(R.id.view_01);btnTest.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View arg0) {//可以看出这里启动了一个线程来操作消息的封装和发送的工作//这样原来主线程的发送就变成了其他线程的发送,简单吧?呵呵new MyThread().start();   }});}class MyHandler extends Handler{public MyHandler(Looper looper){super(looper);}public void handleMessage(Message msg){super.handleMessage(msg);textView.setText("我是主线程的Handler,收到了消息:"+(String)msg.obj);}}//加了一个线程类class MyThread extends Thread{public void run(){Looper looper = Looper.getMainLooper(); //主线程的Looper对象//这里以主线程的Looper对象创建了handler,//所以,这个handler发送的Message会被传递给主线程的MessageQueue。handler = new MyHandler(looper);//构建Message对象//第一个参数:是自己指定的message代号,方便在handler选择性地接收//第二三个参数没有什么意义//第四个参数需要封装的对象Message msg = handler.obtainMessage(1,1,1,"其他线程发消息了");handler.sendMessage(msg); //发送消息           }}
}

3、主线程给其他线程发送Message

package test.message;import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.Looper;
import android.os.Message;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;public class MainActivity extends Activity {private Button btnTest;private TextView textView;private Handler handler;@Overridepublic void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.main);btnTest = (Button)this.findViewById(R.id.btn_01);textView = (TextView)this.findViewById(R.id.view_01);//启动线程new MyThread().start();   btnTest.setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View arg0) {//这里handler的实例化在线程中//线程启动的时候就已经实例化了Message msg = handler.obtainMessage(1,1,1,"主线程发送的消息");handler.sendMessage(msg);}});}class MyHandler extends Handler{public MyHandler(Looper looper){super(looper);}public void handleMessage(Message msg){super.handleMessage(msg);textView.setText("我是主线程的Handler,收到了消息:"+(String)msg.obj);}}class MyThread extends Thread{public void run(){Looper.prepare(); //创建该线程的Looper对象,用于接收消息//注意了:这里的handler是定义在主线程中的哦,呵呵,//前面看到直接使用了handler对象,是不是在找,在什么地方实例化的呢?//现在看到了吧???呵呵,开始的时候实例化不了,因为该线程的Looper对象//还不存在呢。现在可以实例化了//这里Looper.myLooper()获得的就是该线程的Looper对象了handler = new ThreadHandler(Looper.myLooper());//这个方法,有疑惑吗?//其实就是一个循环,循环从MessageQueue中取消息。//不经常去看看,你怎么知道你有新消息呢???Looper.loop();}//定义线程类中的消息处理类class ThreadHandler extends Handler{public ThreadHandler(Looper looper){super(looper);}public void handleMessage(Message msg){//这里对该线程中的MessageQueue中的Message进行处理//这里我们再返回给主线程一个消息handler = new MyHandler(Looper.getMainLooper());Message msg2 = handler.obtainMessage(1,1,1,"子线程收到:"+(String)msg.obj);handler.sendMessage(msg2);}}}
}

4、其他线程给自己发送Message
packagetest.message;import android.app.Activity;
import android.os.Bundle;
import android.os.Handler;
import android.os.Looper;
import android.os.Message;
import android.view.View;
import android.widget.Button;
import android.widget.TextView;public class MainActivity extendsActivity {privateButton btnTest;privateTextView textView;privateHandler handler;@Overridepublicvoid onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.main);btnTest = (Button)this.findViewById(R.id.btn_01);textView = (TextView)this.findViewById(R.id.view_01);btnTest.setOnClickListener(newView.OnClickListener() {@Overridepublicvoid onClick(View arg0) {//启动线程newMyThread().start();   }});}classMyHandler extendsHandler{publicMyHandler(Looper looper){super(looper);}publicvoid handleMessage(Message msg){super.handleMessage(msg);textView.setText((String)msg.obj);}}   classMyThread extendsThread{publicvoid run(){Looper.prepare();//创建该线程的Looper对象//这里Looper.myLooper()获得的就是该线程的Looper对象了handler =new ThreadHandler(Looper.myLooper());Message msg = handler.obtainMessage(1,1,1,"我自己");handler.sendMessage(msg);Looper.loop();}//定义线程类中的消息处理类classThreadHandler extendsHandler{publicThreadHandler(Looper looper){super(looper);}publicvoid handleMessage(Message msg){//这里对该线程中的MessageQueue中的Message进行处理//这里我们再返回给主线程一个消息//加入判断看看是不是该线程自己发的信息if(msg.what ==1 && msg.obj.equals("我自己")){handler =new MyHandler(Looper.getMainLooper());Message msg2 = handler.obtainMessage(1,1,1,"禀告主线程:我收到了自己发给自己的Message");handler.sendMessage(msg2);               }}}}
}


这篇关于Android之Message,MessageQueue,Looper,Handler详解(带示例)的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/575101

相关文章

PHP轻松处理千万行数据的方法详解

《PHP轻松处理千万行数据的方法详解》说到处理大数据集,PHP通常不是第一个想到的语言,但如果你曾经需要处理数百万行数据而不让服务器崩溃或内存耗尽,你就会知道PHP用对了工具有多强大,下面小编就... 目录问题的本质php 中的数据流处理:为什么必不可少生成器:内存高效的迭代方式流量控制:避免系统过载一次性

MySQL的JDBC编程详解

《MySQL的JDBC编程详解》:本文主要介绍MySQL的JDBC编程,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录前言一、前置知识1. 引入依赖2. 认识 url二、JDBC 操作流程1. JDBC 的写操作2. JDBC 的读操作总结前言本文介绍了mysq

Redis 的 SUBSCRIBE命令详解

《Redis的SUBSCRIBE命令详解》Redis的SUBSCRIBE命令用于订阅一个或多个频道,以便接收发送到这些频道的消息,本文给大家介绍Redis的SUBSCRIBE命令,感兴趣的朋友跟随... 目录基本语法工作原理示例消息格式相关命令python 示例Redis 的 SUBSCRIBE 命令用于订

使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解

《使用Python批量将.ncm格式的音频文件转换为.mp3格式的实战详解》本文详细介绍了如何使用Python通过ncmdump工具批量将.ncm音频转换为.mp3的步骤,包括安装、配置ffmpeg环... 目录1. 前言2. 安装 ncmdump3. 实现 .ncm 转 .mp34. 执行过程5. 执行结

Python中 try / except / else / finally 异常处理方法详解

《Python中try/except/else/finally异常处理方法详解》:本文主要介绍Python中try/except/else/finally异常处理方法的相关资料,涵... 目录1. 基本结构2. 各部分的作用tryexceptelsefinally3. 执行流程总结4. 常见用法(1)多个e

Python中logging模块用法示例总结

《Python中logging模块用法示例总结》在Python中logging模块是一个强大的日志记录工具,它允许用户将程序运行期间产生的日志信息输出到控制台或者写入到文件中,:本文主要介绍Pyt... 目录前言一. 基本使用1. 五种日志等级2.  设置报告等级3. 自定义格式4. C语言风格的格式化方法

SpringBoot日志级别与日志分组详解

《SpringBoot日志级别与日志分组详解》文章介绍了日志级别(ALL至OFF)及其作用,说明SpringBoot默认日志级别为INFO,可通过application.properties调整全局或... 目录日志级别1、级别内容2、调整日志级别调整默认日志级别调整指定类的日志级别项目开发过程中,利用日志

Java中的抽象类与abstract 关键字使用详解

《Java中的抽象类与abstract关键字使用详解》:本文主要介绍Java中的抽象类与abstract关键字使用详解,本文通过实例代码给大家介绍的非常详细,感兴趣的朋友跟随小编一起看看吧... 目录一、抽象类的概念二、使用 abstract2.1 修饰类 => 抽象类2.2 修饰方法 => 抽象方法,没有

Spring 中的切面与事务结合使用完整示例

《Spring中的切面与事务结合使用完整示例》本文给大家介绍Spring中的切面与事务结合使用完整示例,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考... 目录 一、前置知识:Spring AOP 与 事务的关系 事务本质上就是一个“切面”二、核心组件三、完

MySQL8 密码强度评估与配置详解

《MySQL8密码强度评估与配置详解》MySQL8默认启用密码强度插件,实施MEDIUM策略(长度8、含数字/字母/特殊字符),支持动态调整与配置文件设置,推荐使用STRONG策略并定期更新密码以提... 目录一、mysql 8 密码强度评估机制1.核心插件:validate_password2.密码策略级