随笔Android IPC进程通信之——Messenger

2024-06-08 13:32

本文主要是介绍随笔Android IPC进程通信之——Messenger,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

随笔Android IPC进程通信之——Messenger

android中用于进程通信和数据共享的方式有多种,最基本的Bundle,文件共享,Messenger,AIDL,ContentProvider,Socket等,他们的区别和使用场景这里就不再说了,详情自行查看了解了解,本文直接介绍方式之一Messenger(信使)的使用方式。

代码为例,开始看看我们遇到的问题,模拟多进程数据同步问题

1.新建一个IPCDemo,创建一个自己的Applciation并声明一个公用变量value

这里写图片描述

2.新建两个Activity,在第一个MainActivity中修改Oncreate中修改 Application中value值为8
这里写图片描述

由MainActivity进入第二个页面(服务Activity)注意此activity在Manifest中配置中添加android:process=”:remote”,目的是让其单独运行在一个进程中,和MainActivity不同进程。
这里写图片描述

在ServerActivity中我们同样打印Applcation中的value;
结果如下:
这里写图片描述

3.引起的问题 我们知道一个应用中Application是一个重量级的,由系统创建的单一实例,其中的数据是共享的,那么这里我们在MainActivity中修改了value值,这个应用中的其他访问的地方应该都是修改后的值,但上面的结果却不是这样。

原因就在于,我们的ServerActivity配置了android:process=":remote",这会导致ServerActivity运行在另一进程中,说白了就是,MainActivity是一个应用,ServerActivity是一个应用,既然是两个不同应用,那么就会有两个不同的Application实例,Application会创建多次(这里两次),这两个实例分别位于不同进程(应用中),这就是为什么修改第一个全局值,而第二个进程中的值没有任何影响的原因。

4.解决问题 我们如何让第一个进程修改变量值的时候,通知另一个进程同步修改对应的值?接下来就利用本文介绍的Messenger。

Messenger 运用

messenger 其实是基于消息的进程通信方式,最基本的理解就是有客户端,服务端如下图:
这里写图片描述

接下来代码介绍:
我们的MainActivity用来充当客户端,ServerActivity充当服务端,因为两个页面完全运行在不同的进程中。

首先,创建一个MessengerService,并在manifest中配置成和ServerActivity中相同的进程。代码如下:

/*** Created by Frosty on 2017/6/13.*//public class    MessengerService extends Service{public static String TAG = MessengerService.class.getSimpleName();public final static int FROM_CLIENT = 1;//来此客户端请求public final static int FROM_SERVICE= 2;//来此服务端请求private Messenger msngerCurrent;//当前类使用(发送消息、接收处理消息)@Overridepublic void onCreate() {super.onCreate();msngerCurrent = new Messenger(new MessengerHandler());}@Nullable@Overridepublic IBinder onBind(Intent intent) {return msngerCurrent.getBinder();}public static class MessengerHandler extends Handler{@Overridepublic void handleMessage(Message msg) {switch (msg.what){case FROM_CLIENT:///1.处理接受消息Log.e(TAG,"收到来自客户端【"+msg.getData().getString("name")+"】的消息:"+msg.getData().getString("content"));MyApplication.value = msg.getData().getInt("value");//2.拿到客户端发送消息附带的Messenger,用它回传消息Messenger msngerReply = msg.replyTo;//封装回传消息Message replyMsg = Message.obtain(null,FROM_SERVICE);Bundle bundle = new Bundle();bundle.putString("name","阿里云");bundle.putString("content","你好,我这边已经同步修改,值为:"+MyApplication.value);replyMsg.setData(bundle);try {msngerReply.send(replyMsg);//回传消息}catch (Exception e){}break;default:break;}}}
}
这里暂时只说明这个服务的功能,利用MessengerHandler处理发送自客户端来的数据,通过msg.what匹配。

所以我们在客户端(MainActivity中去注册绑定这个服务,并在修改appvalue时候发送一个消息给这个服务)

public class MainActivity extends AppCompatActivity {private Messenger msngerCurrent;private class MessengerHandler extends Handler{@Overridepublic void handleMessage(Message msg) {switch (msg.what){case MessengerService.FROM_SERVICE:Log.e("MessengerService","收到来自服务端【"+msg.getData().getString("name")+"】的回应:"+msg.getData().getString("content"));unbindService(serviceConnection);break;}}}@Overrideprotected void onCreate(Bundle savedInstanceState) {super.onCreate(savedInstanceState);setContentView(R.layout.activity_main);MyApplication.value = 8;//此处修改值Log.e("appvalue","MainActivity中获取到的value:"+MyApplication.value);/*绑定服务*/bindRemoteService();msngerCurrent = new Messenger(new MessengerHandler());findViewById(R.id.btn_requestServer).setOnClickListener(new View.OnClickListener() {@Overridepublic void onClick(View v) {Intent intent = new Intent(MainActivity.this,ServerActivity.class);startActivity(intent);}});}///服务链接状态监听private ServiceConnection serviceConnection = new ServiceConnection() {@Overridepublic void onServiceConnected(ComponentName name, IBinder service) {Messenger messengerClient = new Messenger(service);/*封装消息体*/Message msg = Message.obtain(null,MessengerService.FROM_CLIENT);//对应服务端接受码Bundle bundle = new Bundle();bundle.putString("name","张山");bundle.putString("content","我修改了我们的共享值,请你务必修改!");bundle.putInt("value",MyApplication.value);msg.setData(bundle);msg.replyTo=msngerCurrent;//发送try{messengerClient.send(msg);}catch (Exception e){}}@Overridepublic void onServiceDisconnected(ComponentName name) {}};public void bindRemoteService(){Intent intent = new Intent(this, MessengerService.class);bindService(intent,serviceConnection, Context.BIND_AUTO_CREATE);}@Overrideprotected void onDestroy() {unbindService(serviceConnection);super.onDestroy();}
}

我们详细解释一下工作流程
- 我们首先得创建一个ServiceConnection 用于检测服务绑定后或者断开后的操作如下图,绑定服务链接后,回掉调nServiceConnected方法,在这个方法中,我们构造一个客户端Messenger
这里写图片描述并用message封装数据,然后用这个Messenger发送消息即可

发送消息后,MessengerService便可以接收到消息处理,修改当前进程的appvalue值,并回传附加消息给客户端,注意这里回传的Messenger,直接在获取消息体中用msg.replyTo获取,前提是客户端相应的在传过来的msg中replyTo赋值Mesenger,此messenger在客户端也用来处理服务端响应回来的数据。

这里写图片描述

5.结果 我们重启应用,重新打印两个值
这里写图片描述
发现在主进程修改appvalue后,另一个进程获取到的值也及时得到同步了。

End注意:Messenger通信,可使用它进行进程间的通信,而Messenger对Service的请求采用队列的方式,因此它不支持多线程通信。

这篇关于随笔Android IPC进程通信之——Messenger的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Android协程高级用法大全

《Android协程高级用法大全》这篇文章给大家介绍Android协程高级用法大全,本文结合实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友跟随小编一起学习吧... 目录1️⃣ 协程作用域(CoroutineScope)与生命周期绑定Activity/Fragment 中手

Linux系统管理与进程任务管理方式

《Linux系统管理与进程任务管理方式》本文系统讲解Linux管理核心技能,涵盖引导流程、服务控制(Systemd与GRUB2)、进程管理(前台/后台运行、工具使用)、计划任务(at/cron)及常用... 目录引言一、linux系统引导过程与服务控制1.1 系统引导的五个关键阶段1.2 GRUB2的进化优

Android 缓存日志Logcat导出与分析最佳实践

《Android缓存日志Logcat导出与分析最佳实践》本文全面介绍AndroidLogcat缓存日志的导出与分析方法,涵盖按进程、缓冲区类型及日志级别过滤,自动化工具使用,常见问题解决方案和最佳实... 目录android 缓存日志(Logcat)导出与分析全攻略为什么要导出缓存日志?按需过滤导出1. 按

Android Paging 分页加载库使用实践

《AndroidPaging分页加载库使用实践》AndroidPaging库是Jetpack组件的一部分,它提供了一套完整的解决方案来处理大型数据集的分页加载,本文将深入探讨Paging库... 目录前言一、Paging 库概述二、Paging 3 核心组件1. PagingSource2. Pager3.

一文解密Python进行监控进程的黑科技

《一文解密Python进行监控进程的黑科技》在计算机系统管理和应用性能优化中,监控进程的CPU、内存和IO使用率是非常重要的任务,下面我们就来讲讲如何Python写一个简单使用的监控进程的工具吧... 目录准备工作监控CPU使用率监控内存使用率监控IO使用率小工具代码整合在计算机系统管理和应用性能优化中,监

Python实现MQTT通信的示例代码

《Python实现MQTT通信的示例代码》本文主要介绍了Python实现MQTT通信的示例代码,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一... 目录1. 安装paho-mqtt库‌2. 搭建MQTT代理服务器(Broker)‌‌3. pytho

Linux进程CPU绑定优化与实践过程

《Linux进程CPU绑定优化与实践过程》Linux支持进程绑定至特定CPU核心,通过sched_setaffinity系统调用和taskset工具实现,优化缓存效率与上下文切换,提升多核计算性能,适... 目录1. 多核处理器及并行计算概念1.1 多核处理器架构概述1.2 并行计算的含义及重要性1.3 并

Linux下进程的CPU配置与线程绑定过程

《Linux下进程的CPU配置与线程绑定过程》本文介绍Linux系统中基于进程和线程的CPU配置方法,通过taskset命令和pthread库调整亲和力,将进程/线程绑定到特定CPU核心以优化资源分配... 目录1 基于进程的CPU配置1.1 对CPU亲和力的配置1.2 绑定进程到指定CPU核上运行2 基于

Android kotlin中 Channel 和 Flow 的区别和选择使用场景分析

《Androidkotlin中Channel和Flow的区别和选择使用场景分析》Kotlin协程中,Flow是冷数据流,按需触发,适合响应式数据处理;Channel是热数据流,持续发送,支持... 目录一、基本概念界定FlowChannel二、核心特性对比数据生产触发条件生产与消费的关系背压处理机制生命周期

Android ClassLoader加载机制详解

《AndroidClassLoader加载机制详解》Android的ClassLoader负责加载.dex文件,基于双亲委派模型,支持热修复和插件化,需注意类冲突、内存泄漏和兼容性问题,本文给大家介... 目录一、ClassLoader概述1.1 类加载的基本概念1.2 android与Java Class