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

相关文章

性能分析之MySQL索引实战案例

文章目录 一、前言二、准备三、MySQL索引优化四、MySQL 索引知识回顾五、总结 一、前言 在上一讲性能工具之 JProfiler 简单登录案例分析实战中已经发现SQL没有建立索引问题,本文将一起从代码层去分析为什么没有建立索引? 开源ERP项目地址:https://gitee.com/jishenghua/JSH_ERP 二、准备 打开IDEA找到登录请求资源路径位置

JAVA智听未来一站式有声阅读平台听书系统小程序源码

智听未来,一站式有声阅读平台听书系统 🌟 开篇:遇见未来,从“智听”开始 在这个快节奏的时代,你是否渴望在忙碌的间隙,找到一片属于自己的宁静角落?是否梦想着能随时随地,沉浸在知识的海洋,或是故事的奇幻世界里?今天,就让我带你一起探索“智听未来”——这一站式有声阅读平台听书系统,它正悄悄改变着我们的阅读方式,让未来触手可及! 📚 第一站:海量资源,应有尽有 走进“智听

Android实现任意版本设置默认的锁屏壁纸和桌面壁纸(两张壁纸可不一致)

客户有些需求需要设置默认壁纸和锁屏壁纸  在默认情况下 这两个壁纸是相同的  如果需要默认的锁屏壁纸和桌面壁纸不一样 需要额外修改 Android13实现 替换默认桌面壁纸: 将图片文件替换frameworks/base/core/res/res/drawable-nodpi/default_wallpaper.*  (注意不能是bmp格式) 替换默认锁屏壁纸: 将图片资源放入vendo

Android平台播放RTSP流的几种方案探究(VLC VS ExoPlayer VS SmartPlayer)

技术背景 好多开发者需要遴选Android平台RTSP直播播放器的时候,不知道如何选的好,本文针对常用的方案,做个大概的说明: 1. 使用VLC for Android VLC Media Player(VLC多媒体播放器),最初命名为VideoLAN客户端,是VideoLAN品牌产品,是VideoLAN计划的多媒体播放器。它支持众多音频与视频解码器及文件格式,并支持DVD影音光盘,VCD影

Java ArrayList扩容机制 (源码解读)

结论:初始长度为10,若所需长度小于1.5倍原长度,则按照1.5倍扩容。若不够用则按照所需长度扩容。 一. 明确类内部重要变量含义         1:数组默认长度         2:这是一个共享的空数组实例,用于明确创建长度为0时的ArrayList ,比如通过 new ArrayList<>(0),ArrayList 内部的数组 elementData 会指向这个 EMPTY_EL

如何在Visual Studio中调试.NET源码

今天偶然在看别人代码时,发现在他的代码里使用了Any判断List<T>是否为空。 我一般的做法是先判断是否为null,再判断Count。 看了一下Count的源码如下: 1 [__DynamicallyInvokable]2 public int Count3 {4 [__DynamicallyInvokable]5 get

SWAP作物生长模型安装教程、数据制备、敏感性分析、气候变化影响、R模型敏感性分析与贝叶斯优化、Fortran源代码分析、气候数据降尺度与变化影响分析

查看原文>>>全流程SWAP农业模型数据制备、敏感性分析及气候变化影响实践技术应用 SWAP模型是由荷兰瓦赫宁根大学开发的先进农作物模型,它综合考虑了土壤-水分-大气以及植被间的相互作用;是一种描述作物生长过程的一种机理性作物生长模型。它不但运用Richard方程,使其能够精确的模拟土壤中水分的运动,而且耦合了WOFOST作物模型使作物的生长描述更为科学。 本文让更多的科研人员和农业工作者

MOLE 2.5 分析分子通道和孔隙

软件介绍 生物大分子通道和孔隙在生物学中发挥着重要作用,例如在分子识别和酶底物特异性方面。 我们介绍了一种名为 MOLE 2.5 的高级软件工具,该工具旨在分析分子通道和孔隙。 与其他可用软件工具的基准测试表明,MOLE 2.5 相比更快、更强大、功能更丰富。作为一项新功能,MOLE 2.5 可以估算已识别通道的物理化学性质。 软件下载 https://pan.quark.cn/s/57

工厂ERP管理系统实现源码(JAVA)

工厂进销存管理系统是一个集采购管理、仓库管理、生产管理和销售管理于一体的综合解决方案。该系统旨在帮助企业优化流程、提高效率、降低成本,并实时掌握各环节的运营状况。 在采购管理方面,系统能够处理采购订单、供应商管理和采购入库等流程,确保采购过程的透明和高效。仓库管理方面,实现库存的精准管理,包括入库、出库、盘点等操作,确保库存数据的准确性和实时性。 生产管理模块则涵盖了生产计划制定、物料需求计划、

[Linux]:进程(下)

✨✨ 欢迎大家来到贝蒂大讲堂✨✨ 🎈🎈养成好习惯,先赞后看哦~🎈🎈 所属专栏:Linux学习 贝蒂的主页:Betty’s blog 1. 进程终止 1.1 进程退出的场景 进程退出只有以下三种情况: 代码运行完毕,结果正确。代码运行完毕,结果不正确。代码异常终止(进程崩溃)。 1.2 进程退出码 在编程中,我们通常认为main函数是代码的入口,但实际上它只是用户级