热门标签 | HotTags
当前位置:  开发笔记 > 编程语言 > 正文

使用Messenger实现进程间的相互通信

通过Messenger可以在不同进程之间通过传递Message对象实现进程间数据的传递;Messenger是一种轻量级的IPC方案,低层使用AIDL实现。由于Messenger一次处理

通过Messenger可以在不同进程之间通过传递Message对象实现进程间数据的传递;Messenger是一种轻量级的IPC方案,低层使用AIDL实现。由于Messenger一次处理一个请求,因此在不存在并发执行的情况下,不再考虑线程同步问题。
下面我们来看具体的实现
我们这里首先在一个应用中创建不同的两个进程,创建MessengerActivity与MessengerService这两个处于不同进程的类,在MessengerService注册时指定它所对应的进程

<activity android:name="com.example.ipcstudy.MessengerActivity">
<intent-filter>
<action android:name="android.intent.action.MAIN" />

<category android:name="android.intent.category.LAUNCHER" />
intent-filter>
activity>
<service android:name="com.example.ipcstudy.MessengerService"
android:process=":MessengerService"/>
"

android:process=”:MessengerService” 指定MessengerService在com.example.ipcstudy:MessengerService进程中;MessengerActivity默认在com.example.ipcstudy进程中。

MessengerService类的实现:

package com.example.ipcstudy;

import android.app.Service;
import android.content.Intent;
import android.os.Bundle;
import android.os.Handler;
import android.os.IBinder;
import android.os.Message;
import android.os.Messenger;
import android.os.RemoteException;
import android.widget.Toast;

public class MessengerService extends Service{
//创建Handler用于MessengerActivity中传递的Message对象
private Handler messengerHandler=new Handler(){
public void handleMessage(android.os.Message msg) {
switch (msg.what) {
case MessengerActivity.MESSENGER:
//获取信息内容并Toast
Toast.makeText(MessengerService.this, msg.getData().getString("content"), 0).show();
//获取MessengerActivity中replyMessenger对象
Messenger messenger=msg.replyTo;
//创建传递的消息对象
Message replyMessage=Message.obtain(null, MessengerActivity.MESSENGER);
Bundle bundle=new Bundle();
bundle.putString("reply", "我已经收到消息!");
replyMessage.setData(bundle);
try {
//给MessengerActivity发送消息
messenger.send(replyMessage);
} catch (RemoteException e) {
e.printStackTrace();
}
break;
default:
break;
}
};
};
private Messenger messenger=new Messenger(messengerHandler);
@Override
public IBinder onBind(Intent intent) {
//返回IBinder对象
return messenger.getBinder();
}

}

MessengerActivity类的实现

package com.example.ipcstudy;

import android.app.Activity;
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.Message;
import android.os.Messenger;
import android.os.RemoteException;
import android.widget.Toast;

public class MessengerActivity extends Activity{
public static final int MESSENGER=111;
private ServiceConnection cOnn=new ServiceConnection() {
@Override
public void onServiceDisconnected(ComponentName name) {}
@Override
public void onServiceConnected(ComponentName name, IBinder service) {
//通过服务返回的IBinder对象创建Messenger对象
Messenger messenger=new Messenger(service);
//创建Message对象
Message message=new Message();
message.what=MESSENGER;
Bundle bundle=new Bundle();
bundle.putString("content", "通过Messenger进行跨进程通信");
message.setData(bundle);
//将接收消息的replyMessenger传递给MessengerService
message.replyTo=replyMessenger;
try {
//向MessengerService发送消息
messenger.send(message);
} catch (RemoteException e) {
e.printStackTrace();
}
}
};
/**
* 用于接收MessengerService传递过来的数据Message对象
*/

private Messenger replyMessenger=new Messenger(new Handler(){
public void handleMessage(Message msg) {
switch (msg.what) {
case MESSENGER:
//获取信息内容并Toast
Toast.makeText(MessengerActivity.this, msg.getData().getString("reply"), 0).show();
break;
default:
break;
}
};
});
@Override
protected void onCreate(Bundle savedInstanceState) {
super.onCreate(savedInstanceState);
//绑定服务
Intent intent=new Intent(this,MessengerService.class);
bindService(intent, conn, Context.BIND_AUTO_CREATE);
}
@Override
protected void onDestroy() {
super.onDestroy();
unbindService(conn);
}
}

注意:
Messenger中进行数据传递必须将数据放入Message中,而Messenger与Message都实现了Parcelable接口,因此可以跨进程传输。
Message只能使用what,arg1,arg2,Bundle,replyto传递信息,Message中object字段仅支持系统提供的实现了Parcelable接口的对象传递信息,我们自己定义Parcelable对象不能进行传输。


推荐阅读
  • 本文介绍如何利用动态规划算法解决经典的0-1背包问题。通过具体实例和代码实现,详细解释了在给定容量的背包中选择若干物品以最大化总价值的过程。 ... [详细]
  • golang常用库:配置文件解析库/管理工具viper使用
    golang常用库:配置文件解析库管理工具-viper使用-一、viper简介viper配置管理解析库,是由大神SteveFrancia开发,他在google领导着golang的 ... [详细]
  • 优化ListView性能
    本文深入探讨了如何通过多种技术手段优化ListView的性能,包括视图复用、ViewHolder模式、分批加载数据、图片优化及内存管理等。这些方法能够显著提升应用的响应速度和用户体验。 ... [详细]
  • 本文详细介绍了 GWT 中 PopupPanel 类的 onKeyDownPreview 方法,提供了多个代码示例及应用场景,帮助开发者更好地理解和使用该方法。 ... [详细]
  • 本文将介绍如何编写一些有趣的VBScript脚本,这些脚本可以在朋友之间进行无害的恶作剧。通过简单的代码示例,帮助您了解VBScript的基本语法和功能。 ... [详细]
  • 本题探讨了一种字符串变换方法,旨在判断两个给定的字符串是否可以通过特定的字母替换和位置交换操作相互转换。核心在于找到这些变换中的不变量,从而确定转换的可能性。 ... [详细]
  • Explore a common issue encountered when implementing an OAuth 1.0a API, specifically the inability to encode null objects and how to resolve it. ... [详细]
  • 本文将介绍如何使用 Go 语言编写和运行一个简单的“Hello, World!”程序。内容涵盖开发环境配置、代码结构解析及执行步骤。 ... [详细]
  • 1:有如下一段程序:packagea.b.c;publicclassTest{privatestaticinti0;publicintgetNext(){return ... [详细]
  • 本文详细介绍了如何在Linux系统上安装和配置Smokeping,以实现对网络链路质量的实时监控。通过详细的步骤和必要的依赖包安装,确保用户能够顺利完成部署并优化其网络性能监控。 ... [详细]
  • 本文介绍了Java并发库中的阻塞队列(BlockingQueue)及其典型应用场景。通过具体实例,展示了如何利用LinkedBlockingQueue实现线程间高效、安全的数据传递,并结合线程池和原子类优化性能。 ... [详细]
  • 1.如何在运行状态查看源代码?查看函数的源代码,我们通常会使用IDE来完成。比如在PyCharm中,你可以Ctrl+鼠标点击进入函数的源代码。那如果没有IDE呢?当我们想使用一个函 ... [详细]
  • 本文详细介绍 Go+ 编程语言中的上下文处理机制,涵盖其基本概念、关键方法及应用场景。Go+ 是一门结合了 Go 的高效工程开发特性和 Python 数据科学功能的编程语言。 ... [详细]
  • 探讨如何通过编程技术实现100个并发连接,解决线程创建顺序问题,并提供高效的并发测试方案。 ... [详细]
  • 本文介绍如何使用Objective-C结合dispatch库进行并发编程,以提高素数计数任务的效率。通过对比纯C代码与引入并发机制后的代码,展示dispatch库的强大功能。 ... [详细]
author-avatar
主持人谷佳霓期_426
这个家伙很懒,什么也没留下!
PHP1.CN | 中国最专业的PHP中文社区 | DevBox开发工具箱 | json解析格式化 |PHP资讯 | PHP教程 | 数据库技术 | 服务器技术 | 前端开发技术 | PHP框架 | 开发工具 | 在线工具
Copyright © 1998 - 2020 PHP1.CN. All Rights Reserved | 京公网安备 11010802041100号 | 京ICP备19059560号-4 | PHP1.CN 第一PHP社区 版权所有