随笔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、动态请求权限2、悬浮窗权限说明3、检查动态权限4、申请动态权限5、权限设置完毕后

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

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

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

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

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中的模板一、前言在项目开发过程中,尤

python多进程实现数据共享的示例代码

《python多进程实现数据共享的示例代码》本文介绍了Python中多进程实现数据共享的方法,包括使用multiprocessing模块和manager模块这两种方法,具有一定的参考价值,感兴趣的可以... 目录背景进程、进程创建进程间通信 进程间共享数据共享list实践背景 安卓ui自动化框架,使用的是

Android数据库Room的实际使用过程总结

《Android数据库Room的实际使用过程总结》这篇文章主要给大家介绍了关于Android数据库Room的实际使用过程,详细介绍了如何创建实体类、数据访问对象(DAO)和数据库抽象类,需要的朋友可以... 目录前言一、Room的基本使用1.项目配置2.创建实体类(Entity)3.创建数据访问对象(DAO

C#如何优雅地取消进程的执行之Cancellation详解

《C#如何优雅地取消进程的执行之Cancellation详解》本文介绍了.NET框架中的取消协作模型,包括CancellationToken的使用、取消请求的发送和接收、以及如何处理取消事件... 目录概述与取消线程相关的类型代码举例操作取消vs对象取消监听并响应取消请求轮询监听通过回调注册进行监听使用Wa

Android WebView的加载超时处理方案

《AndroidWebView的加载超时处理方案》在Android开发中,WebView是一个常用的组件,用于在应用中嵌入网页,然而,当网络状况不佳或页面加载过慢时,用户可能会遇到加载超时的问题,本... 目录引言一、WebView加载超时的原因二、加载超时处理方案1. 使用Handler和Timer进行超