Android 进程间通信(四) - Messenger 以及源码分析

2024-06-07 20:08

本文主要是介绍Android 进程间通信(四) - Messenger 以及源码分析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

在上几篇文章中,我们已经学习了Binder 这种通信模式,这章中,我们来学习 Messenger;

Messenger可以翻译为信使,它可以在不同进程间传递 Message 对象。Messenger 是一种轻量的 IPC 防范,它的底层实现是 AIDL,后面我们会分析它的源码;

Messenger 一次只能处理一个请求,因此并不适合高并发的问题,也不用考虑线程同步的问题,因为服务端中不存在并发执行的情形。

一. 实现Messenger通信

从上面可以知道,可以传递 Message,一些简单的数据,我们就可以通过这种方式去实现了。下面我们一起来实现它。
今天的效果是,客户端发送两个数字,服务端返回两数之和:
在这里插入图片描述

1.1. 服务端

在服务端中,我们可以直接 new 一个 Messenger ,传递一个 Handler,如下:

public class MessengerService extends Service {private static final String TAG = "MessageService";public MessengerService() {}Messenger mMessenger = new Messenger(new Handler(Looper.getMainLooper()){@Overridepublic void handleMessage(@NonNull Message msg) {super.handleMessage(msg);}});@Overridepublic IBinder onBind(Intent intent) {return mMessenger.getBinder();}
}

可以看到,Messenger 的构造方法中,传递了一个 Handler,当接收到客户端的 Message 消息时,会在 handleMessage() 方法中处理,并通过 mMessenger.getBinder() 返回 binder 对象。

接着,需要通过 Message 拿到两个数字,并返回相加的结果,如下:

        @Overridepublic void handleMessage(@NonNull Message msg) {super.handleMessage(msg);try {int num = msg.arg1 + msg.arg2;Message message = Message.obtain();message.what = 2;message.arg1 = num;//通过客户端的 Messenger 来把数据发送回去msg.replyTo.send(message);} catch (RemoteException e) {e.printStackTrace();}}

需要注意的是,数据在返回的时候,需要通过 msg.replyTo 去发送,它的对象为 Messenger 。
这样,服务端就写好了。

1.2. 客户端

接着,我们新建一个moudle,通过包名加类名的方式去 bind 这个服务。如下:

Intent intent = new Intent();
//绑定 服务
intent.setClassName("com.example.ipcdemo","com.example.ipcdemo.service.MessengerService");
mRemoteService = new RemoteService();
bindService(intent,mRemoteService, Service.BIND_AUTO_CREATE);

在连接的时候,创建 Messenger :

    class RemoteService implements ServiceConnection {@Overridepublic void onServiceConnected(ComponentName name, IBinder service) {mBinder = new Messenger(service);mSb.append("连接上服务端").append("\n");mTextView.setText(mSb.toString());}@Overridepublic void onServiceDisconnected(ComponentName name) {mBinder = null;}}

这样,我们就拿到了服务端mBinder的实例了,然后通过点击事件,把数据发送过去:

    public void testBtn(View view) {Message msg = Message.obtain();msg.arg1 = (int) (Math.random() * 100+1);msg.arg2 = (int) (Math.random() * 100+1);mTextView.setText(mSb.toString());//传递 client 的 messenger ,实现服务端与客户端通信msg.replyTo = mMessenger;if (mBinder != null) {try {mBinder.send(msg);} catch (RemoteException e) {e.printStackTrace();}}}

同时,也要注意,把要接受服务端信息的 Messenger 绑定到 msg.replyTo 中。客户端的 Messenger 如下:

    Messenger mMessenger = new Messenger(new Handler(Looper.getMainLooper()){@Overridepublic void handleMessage(@NonNull Message msg) {super.handleMessage(msg);mSb.append(msg.arg1).append("\n");mTextView.setText(mSb.toString());}});

这样,我们就实现所有功能了。效果如下面的 gif 。

使用 Messenger 的优缺点如下:

优点缺点使用场景
功能一般,支持一对多穿行通信,支持实时通信不能很好的处理高并发情形,不支持RPC,数据通过Message传输,因此只能传输Bundle支持的数据类型,比如 msg.setData()低并发的一对多及即时通信,无RPC需求,或者无需要返回结果的RPC需求

二. 源码分析

上面说到,Messenger 是基于 AIDL 的,怎么得出来的呢?它的构造方法传递要求传入的 Handler,进入到里面看一下:

    public Messenger(Handler target) {mTarget = target.getIMessenger();}

然后,去到 Handler ,找到 getIMessage() 方法:
在这里插入图片描述
可以看到,mMessenger 的实现类为 MessengerImpl ,而它继承自 IMessenger.Stub ;看到这里有没有很熟悉? 没错,跟我们的 AIDL 的实现类一样,通过源码也知道,IMessenger 确实是个 AIDL 接口:

package android.os;import android.os.Message;/** @hide */
oneway interface IMessenger {void send(in Message msg);
}

在它的 send 方法中,把数据又通过 Handler 的 sendMessage 方法,发送给自身,从而当有数据来的时候,就会调用它的 handleMessage()方法。

而在客户端的绑定数据中,也验证了这个问题,比如:
在这里插入图片描述
去到构造方法中,可以看到:
在这里插入图片描述
通过 asInterface 拿到binder。

这样,我们就了解清楚了。

这篇关于Android 进程间通信(四) - Messenger 以及源码分析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Android 悬浮窗开发示例((动态权限请求 | 前台服务和通知 | 悬浮窗创建 )

《Android悬浮窗开发示例((动态权限请求|前台服务和通知|悬浮窗创建)》本文介绍了Android悬浮窗的实现效果,包括动态权限请求、前台服务和通知的使用,悬浮窗权限需要动态申请并引导... 目录一、悬浮窗 动态权限请求1、动态请求权限2、悬浮窗权限说明3、检查动态权限4、申请动态权限5、权限设置完毕后

Springboot中分析SQL性能的两种方式详解

《Springboot中分析SQL性能的两种方式详解》文章介绍了SQL性能分析的两种方式:MyBatis-Plus性能分析插件和p6spy框架,MyBatis-Plus插件配置简单,适用于开发和测试环... 目录SQL性能分析的两种方式:功能介绍实现方式:实现步骤:SQL性能分析的两种方式:功能介绍记录

Android里面的Service种类以及启动方式

《Android里面的Service种类以及启动方式》Android中的Service分为前台服务和后台服务,前台服务需要亮身份牌并显示通知,后台服务则有启动方式选择,包括startService和b... 目录一句话总结:一、Service 的两种类型:1. 前台服务(必须亮身份牌)2. 后台服务(偷偷干

最长公共子序列问题的深度分析与Java实现方式

《最长公共子序列问题的深度分析与Java实现方式》本文详细介绍了最长公共子序列(LCS)问题,包括其概念、暴力解法、动态规划解法,并提供了Java代码实现,暴力解法虽然简单,但在大数据处理中效率较低,... 目录最长公共子序列问题概述问题理解与示例分析暴力解法思路与示例代码动态规划解法DP 表的构建与意义动

linux进程D状态的解决思路分享

《linux进程D状态的解决思路分享》在Linux系统中,进程在内核模式下等待I/O完成时会进入不间断睡眠状态(D状态),这种状态下,进程无法通过普通方式被杀死,本文通过实验模拟了这种状态,并分析了如... 目录1. 问题描述2. 问题分析3. 实验模拟3.1 使用losetup创建一个卷作为pv的磁盘3.

C#使用DeepSeek API实现自然语言处理,文本分类和情感分析

《C#使用DeepSeekAPI实现自然语言处理,文本分类和情感分析》在C#中使用DeepSeekAPI可以实现多种功能,例如自然语言处理、文本分类、情感分析等,本文主要为大家介绍了具体实现步骤,... 目录准备工作文本生成文本分类问答系统代码生成翻译功能文本摘要文本校对图像描述生成总结在C#中使用Deep

Linux环境变量&&进程地址空间详解

《Linux环境变量&&进程地址空间详解》本文介绍了Linux环境变量、命令行参数、进程地址空间以及Linux内核进程调度队列的相关知识,环境变量是系统运行环境的参数,命令行参数用于传递给程序的参数,... 目录一、初步认识环境变量1.1常见的环境变量1.2环境变量的基本概念二、命令行参数2.1通过命令编程

Linux之进程状态&&进程优先级详解

《Linux之进程状态&&进程优先级详解》文章介绍了操作系统中进程的状态,包括运行状态、阻塞状态和挂起状态,并详细解释了Linux下进程的具体状态及其管理,此外,文章还讨论了进程的优先级、查看和修改进... 目录一、操作系统的进程状态1.1运行状态1.2阻塞状态1.3挂起二、linux下具体的状态三、进程的

Android kotlin语言实现删除文件的解决方案

《Androidkotlin语言实现删除文件的解决方案》:本文主要介绍Androidkotlin语言实现删除文件的解决方案,在项目开发过程中,尤其是需要跨平台协作的项目,那么删除用户指定的文件的... 目录一、前言二、适用环境三、模板内容1.权限申请2.Activity中的模板一、前言在项目开发过程中,尤

Go中sync.Once源码的深度讲解

《Go中sync.Once源码的深度讲解》sync.Once是Go语言标准库中的一个同步原语,用于确保某个操作只执行一次,本文将从源码出发为大家详细介绍一下sync.Once的具体使用,x希望对大家有... 目录概念简单示例源码解读总结概念sync.Once是Go语言标准库中的一个同步原语,用于确保某个操