使用 Messenger 跨进程通信

什么是Messenger

Messenger 也是IPC的方案之一,是基于消息的跨进程通信。基于消息是什么意思?Handler是我们最常用的消息机制,所以 Messenger 对于使用者来说就像是使用 Handler。实际上 Messenger 就是 AIDL 的上层封装而已,它们的底层实现原理都是基于 Binder 的。

Messenger的使用

服务端

import android.app.Service
import android.content.Intent
import android.os.Bundle
import android.os.Handler
import android.os.IBinder
import android.os.Looper
import android.os.Message
import android.os.Messenger
import android.util.Logconst val MSG_CLIENT = 0x110
const val MSG_SERVER = 0x111
class MessengerService: Service() {private val mMessenger = Messenger(object : Handler(Looper.getMainLooper()){override fun handleMessage(msg: Message) {Log.d("MessengerService", "currentThread ->" + Thread.currentThread().name)when(msg.what) {MSG_CLIENT -> {// 除非就是一个简单的整型,可以把值放到 arg 等属性上,否则最好使用 bundleval bundle = msg.dataLog.d("MessengerService", "name=${bundle.get("name")}; age=${bundle.get("age")}; height=${bundle.get("height")}")// 服务端封装msgval replyMsg = Message.obtain()replyMsg.what = MSG_SERVERval bundleToC = Bundle()bundleToC.putString("msg",  "我收到了Client的消息")replyMsg.data = bundleToC// 服务端发送给客户端,这里的replyTo是客户端的messenger实例msg.replyTo.send(replyMsg)}}super.handleMessage(msg)}});override fun onBind(intent: Intent?): IBinder? {return mMessenger.binder}
}

Messenger 本身就是 AIDL 的封装,因此还是通过 serviceonBind() 方法里返回 mMessenger.binder。然后它就像是使用 Handler 一样在 handleMessage() 方法内接收 msg。同样的需要在 AndroidManifest.xml 文件中声明该 service

<service android:name=".MessengerService"android:exported="true"><intent-filter><action android:name="com.example.messengerservice"></action></intent-filter>
</service>

客户端

在客户端第一步还是要绑定 Service

import android.content.ComponentName
import android.content.Context
import android.content.Intent
import android.content.ServiceConnection
import android.os.Bundle
import android.os.Handler
import android.os.IBinder
import android.os.Looper
import android.os.Message
import android.os.Messenger
import android.os.RemoteException
import android.util.Log
import androidx.activity.ComponentActivityclass MainActivity: ComponentActivity() {override fun onCreate(savedInstanceState: Bundle?) {super.onCreate(savedInstanceState)bindService()}var mMessenger: Messenger? = nullprivate val mServiceConnection: ServiceConnection = object : ServiceConnection {override fun onServiceConnected(name: ComponentName, service: IBinder) {mMessenger = Messenger(service)Log.i("MainActivity", "onServiceConnected IBinder = $service")}override fun onServiceDisconnected(name: ComponentName) {mMessenger = null}}private fun bindService() {val intent = Intent()intent.component = ComponentName("com.example.interprocess", "com.example.service.MessengerService")bindService(intent, mServiceConnection, Context.BIND_AUTO_CREATE)}fun send() {mMessenger?.let { messenger ->val msg = Message.obtain()msg.what = MSG_CLIENTval bundle = Bundle().apply {putString("name", "张三")putInt("age", 22)putString("sex", "男")putString("height", "175cm")}msg.data = bundlemsg.replyTo = mClientMessengertry {messenger.send(msg)} catch (e: RemoteException) {e.printStackTrace()}}}private val mClientMessenger = Messenger(object : Handler(Looper.getMainLooper()) {override fun handleMessage(msg: Message) {when(msg.what){MSG_SERVER -> {Log.d("MainActivity", "currentThread ->" + Thread.currentThread().name)Log.d("MainActivity", "server callback = " + msg.data.getString("msg"))}}super.handleMessage(msg)}})override fun onDestroy() {super.onDestroy()unbindService(mServiceConnection)}
}

AIDL 使用唯一的差异是将 onServiceConnected()IBinder 参数来构建一个 Messenger 实例。

最后 Messenger 是通过 send() 方法把 msg 发送出去的,感觉就好像是使用 Handler 发送了消息一样, 真正的数据需要保存到 Bundle 里,这里需要提一点的是如果想给 Bundle 里塞实现了 Parcelable 的对象,会在服务端接受参数时爆出 ClassNotFoundException, 因为两个 App 的 Bundle 是不同类加载器加载的,所以我们使用的时候还是把基本类型数据塞到 bundle 里,这对于大量的复杂数据并不是一个好方式。

因此基于消息机制的 Messenger 虽然给我们带来了比 AIDL 更便捷的使用和理解,但并不适合管理复杂数据的跨进程通信。

可以看到上面的代码还给msg.replyTo赋值了一个 mClientMessenger , 这个就是客户端接受消息的 messenger,还记得在服务端回调给客户端的操作就是 msg.replyTo.send(replyMsg),客户端的 messenger 消息处理也是一样的。

我们在服务端和客户端都打印了一下线程,最后发现接受消息时都已经处于主线程了,这和 messenger 构造函数的 handler 入参有关,handler 指定了主线程的 Looper,那么接收消息时就是在主线程。

Messenger 原理

在创建 Messenger 时总是会传入一个 Handler 实例:

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

再进入 Handler 看下:

final IMessenger getIMessenger() {synchronized (mQueue) {if (mMessenger != null) {return mMessenger;}mMessenger = new MessengerImpl();return mMessenger;}
}private final class MessengerImpl extends IMessenger.Stub {public void send(Message msg) {msg.sendingUid = Binder.getCallingUid();Handler.this.sendMessage(msg);}
} 

Handler 里的 MessengerImpl 继承了 IMessenger.Stub,看下 IMessenger

// IMessenger.aidl
oneway interface IMessenger {void send(in Message msg);
}

这完全就是一个标准的 aidl 接口,只是系统帮我们做了这一层的封装,然后通过 Handler 转化为我们熟悉的消息机制。值得注意这里有 oneway 关键字,说明了 Messenger 就是异步调用,因此 send() 方法不需要返回值。 还有就是参数 Message,既然它能跨进程传递,那么它一定是实现了 Parcelable 接口了,且声明了 aidl 接口:

// 包路径:/frameworks/base/core/java/android/os/Message.aidl
package android.os;
parcelable Message;// 包路径:/frameworks/base/core/java/android/os/Message.java
public final class Message implements Parcelable {//...
}

还有一个点,Messenger 实现相互通信的方式是设置一个 msg.replyTo 这个变量也是一个 Messenger, 还记得 AIDL 介绍过支持 aidl 接口的传递,所以客户端的 mClientMessenger 能够传递给服务端,然后调用 send() 方法即可回传数据。

总结

  • Messenger 是基于消息机制的跨进程通信,原理就是对 AIDL 的封装。

  • Messenger不太适合管理复杂数据的跨进程传递。

  • Messenger 天然支持异步的跨进程调用。

  • 无论是客户端还是服务端接收跨进程消息时都处于主线程,免去了切换线程步骤。

注:本文内容转载自 Messenger 跨进程通信

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/80311.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

微服务全栈:深入核心组件与开发技巧

文章目录 1.服务注册与发现1.1. 客户端注册 (ZooKeeper)1.2. 第三方注册 (独立的服务Registrar)1.3. 客户端发现1.4. 服务端发现1.5. Consul1.6. Eureka1.7. SmartStack1.8. Etcd 2. API 网关2.1. 请求转发2.2. 响应合并2.3. 协议转换2.4. 数据转换2.5. 安全认证 3. 配置中心3.…

什么是交换分区以及如何创建交换分区

介绍 交换分区是Linux中的一项功能,可提供虚拟内存空间和多种好处。它允许操作系统有效地处理内存需求。因此,交换分区提高了系统稳定性、响应能力和繁重工作负载处理。 本指南将探讨交换分区及其优缺点,并概述在 Linux 系统上创建和管理交换分区的步骤。 先决条件 运行 …

PHP自己的框架2.0结合容器技术(重构篇二)

目录 1、使用容器实现框架加载类运行 2、 创建框架容器类core/fm/Di.php 3、框架使用容器类来执行public/index.php 4、运行效果还是一样 1、使用容器实现框架加载类运行 2、 创建框架容器类core/fm/Di.php 什么是容器&#xff1f;容器就相当于盒子&#xff0c;把很多类放里…

NTIRE2023-RTSR-Track2 冠军方案详解

编辑 | Happy 首发 | AIWalker 链接 | https://mp.weixin.qq.com/s/JQ5g9yn_OdjR8hi_tWc4jA arXiv:2305.02126 , cvpr2023 , code , video 本文了一个实时&轻量图像超分方案Bicubic&#xff0c;它通过下采样模块降低图像分辨率以减少计算量&#xff0c;在网络尾部采用…

三分钟使用ngrok实现内网穿透

1.官网注册 官网地址&#xff1a;https://ngrok.com/ tips:若使用邮箱注册自行认证 2.下载对应部署电脑 压缩包&#xff08;此处笔者使用自己电脑因此以Windows11作为案例&#xff09; 解压下载的ngrok压缩包&#xff0c;在对应目录进入命令提示符装口&#xff08;也可直接…

免费api接口:物流api,企业工商查询api,游戏api。。。

免费api接口&#xff0c;物流api,企业工商查询api,游戏api。。。都有。 Facebook Games Services - Facebook Games Services 为游戏开发者提供了各种服务, 包括(但不限于) 成就 API, 分数 API, 应用通知, 请求, 游戏养成和 Facebook SDK for Unity.Google Play Games Service…

Git diff 使用 vimdiff 对比差异

在Ubuntu中使用Git时&#xff0c;可使用命令行的git diff命令来对比两次提交的差异&#xff0c;但是这种对比查看方式无法直观地查看修改的差异&#xff0c;在对比和查看时不太方便。 可以使用vimdiff作为Git diff的对比工具&#xff0c;这样就方便了许多&#xff0c;Git的配置…

TensorFlow 03(Keras)

一、tf.keras tf.keras是TensorFlow 2.0的高阶API接口&#xff0c;为TensorFlow的代码提供了新的风格和设计模式&#xff0c;大大提升了TF代码的简洁性和复用性&#xff0c;官方也推荐使用tf.keras来进行模型设计和开发。 1.1 tf.keras中常用模块 如下表所示: 1.2 常用方法 …

TCP/IP网络江湖——数据链路层的协议与传承(数据链路层中篇:数据链路层的协议与帧)

0、引言 网络江湖,宛如千年武林,承载着代代传承的文化和传统。在这个广袤的江湖之中,数据链路层犹如武林门派,代代传承着网络通信的精华。这一部分将带领我们深入探讨数据链路层的协议与传承,揭示其在网络江湖中的精彩故事。 就如同江湖中的武者需要基本功夫作为修行的基础…

mysql如何实现根据经纬度判断某一个坐标是否在一个多边形区域范围内

要根据经纬度判断一个坐标是否在一个多边形区域内&#xff0c;MySQL提供了几种函数来处理地理空间数据&#xff0c;其中包括用于处理多边形区域的函数。 1.创建一个包含多边形区域的表&#xff1a; 首先&#xff0c;创建一个表来存储多边形区域。可以使用ST_GeomFromText函数将…

kuiper 规则sql写法

创建规则对接收到的报文数据进行业务过滤&#xff0c;报文有各种结构的&#xff0c;下面对各种结构报文sql过滤使用进行说明 下面sql规则统一对temperature大于20的数据进行过滤 1&#xff1a;单层结构报文 {"temperature": 35,"humidity": 66 } sql写…

【Leetcode Sheet】Weekly Practice 6

Leetcode Test 2605 从两个数字数组里生成最小数字(9.5) 给你两个只包含 1 到 9 之间数字的数组 nums1 和 nums2 &#xff0c;每个数组中的元素 互不相同 &#xff0c;请你返回 最小 的数字&#xff0c;两个数组都 至少 包含这个数字的某个数位。 提示&#xff1a; 1 < …

MySQL8--my.cnf配置文件的设置

原文网址&#xff1a;MySQL8--my.cfg配置文件的设置_IT利刃出鞘的博客-CSDN博客 简介 本文介绍MySQL8的my.cnf的配置。 典型配置 [client] default-character-setutf8mb4[mysql] default-character-setutf8mb4[mysqld] #服务端口号 默认3306 port3306datadir /work/docker…

一个FlutterCocoapods项目打包问题集锦

一个Flutter&Cocoapods项目打包问题集锦 问题1 github加速问题 cocoapods项目需要访问https://github.com/CocoaPods/Specs.git&#xff0c;众所周知&#xff0c;github经常被墙&#xff0c;导致经常需要借助加速来下载和访问&#xff0c;这里可以使用油猴脚本或者Fastgi…

kibana报错内存溢出问题解决

一、背景&#xff1a; kibana内存溢出&#xff0c;进程被kill掉&#xff0c;导致前端页面访问不到。 报错内容 二、报错原因&#xff1a; 发现是前端 js 报的内存 oom 异常&#xff0c;通过网上资料发现node.js 的默认内存大小为1.4G Node 中通过 JavaScript 使用内存时只能…

Promethues(五)查询-PromQL 语言-保证易懂好学

文章目录 一、介绍二、PromQL 数据类型三、常量1 字符串2 浮点 四、时间序列选择器 Time series Selectors1 即时矢量&#xff08;Instant vector&#xff09;选择器2 范围矢量选择器2.1 时间长度2.2 偏移修饰符2.3 修饰符 3 避免慢速查询和过载 五、子查询六、操作符 Operato…

【C++】深拷贝和浅拷贝 ② ( 默认拷贝构造函数是浅拷贝 | 代码示例 - 浅拷贝造成的问题 )

文章目录 一、默认拷贝构造函数是浅拷贝1、默认拷贝构造函数2、默认拷贝构造函数是浅拷贝机制 二、代码示例 - 浅拷贝造成的问题 一、默认拷贝构造函数是浅拷贝 1、默认拷贝构造函数 如果 C 类中 没有定义拷贝构造函数 , C 编译器会自动为该类提供一个 " 默认的拷贝构造函…

连接MySQL时报错:Public Key Retrieval is not allowed的解决方法

问题描述&#xff1a; DBeaver 连接 mysql 时报错&#xff1a;Public Key Retrieval is not allowed&#xff08;不允许公钥检索&#xff09; 解决方法&#xff1a; 连接设置 -> 驱动属性 -> allowPublicKeyRetrievalfalse&#xff08;这里的运输公钥检索是默认关闭的&a…

如何在RK3568开发板上实现USBNET?——飞凌嵌入式/USB Gadget/USB-NET/网络

本文将借助飞凌嵌入式OK3568-C开发板为大家介绍实现USBNET模式的方法&#xff0c;在这之前需要先知道什么是USB Gadget——USB Gadget是指所开发的电子设备以USB从设备的模式通过USB连接到主机。举个例子&#xff1a;将手机通过USB线插入PC后&#xff0c;手机就是USB Gadget。同…

pt24django教程

静态文件访问 不能与服务器端做动态交互的文件都是静态文件&#xff0c;如: 图片,css,js,音频,视频,html文件(部分) 静态文件配置 在 settings.py 中配置一下两项内容: STATIC_URL 静态文件的访问路径&#xff0c;通过哪个url地址找静态文件 &#xff0c;STATIC_URL ‘/s…