目前面对大多数的需要在异构系统间进行消息传递技术路线,大多会选择socket或webservice。这两种技术的共同特点是耦合紧,调试依赖双方同步,但是效率高。除此以外,使用消息队列(MQ)的应用场景也偶尔能遇到。本文就将要从AMQP协议说起,重点介绍利用RabbitMQ实现C++和Java跨系统开发的实践。

一、AMQP是什么

AMQP又称为高级消息队列协议,是一种进程间进行异步消息的网络协议。它的出现是为了让各类消息中间件提供统一服务,以降低系统集成的开销。目前,完全准寻AMQP协议的消息中间件只有RabbitMQ。虽然各大中间件产品也都针对不同的语言推出了客户端。但是,无论是从业务适应性还是集成通用性上来说,比较推荐的还是RabbitMQ。不同的消息中间件在性能上的差异网上资料很多,这里不再赘述。

amqp协议和http协议一样都是建立在TCP/IP协议簇之上的应用层协议。不同于http协议的,它是一个二进制协议,具有多信道,异步,高效等特点。amqp协议规定了从消息发布者到消息接收者之间的消息传递方式,并且提出了交换机(Exchange)队列(Queue)以及他们之间的路由(Routing)。

作为一套标准协议,使用者甚至可以完全根据amqp的协议规范定制化的开发出客户端和RabbitMQ通信,这一特点也让RabbitMQ在业务通用性上具备了得天独厚的优势。标准的amqp协议格式如下:

amqp://<username>:<password>@<host>:<port>/<virtual>

username: 用户名

password: 登录密码

host: 服务所在主机地址

port: 服务端口号

virtual: 虚拟路径

AMQP协议最值得学习的地方在于,它定义了消息的发送和投递过程:

  交换机(Exchange)负责接收消息,并根据提前指定的规则(Routing)投送消息到特定队列(Queue)。消费者监听队列,并处理消息。如果多个消费者监听同一个队列,消息一般会轮流的发送给它们。以实现负载均衡。此外,通过虚拟路径约束还允许在不同的虚拟路径下建立同命队列。

AMQP协议默认提供了四种类型的交换机:

直接交换机(Direct Exchange):根据路由键的不同将消息直接发送到不同队列,未匹配路由键的消息会被丢弃。

扇形交换机(Funout Exchange):扇形交换机是实现广播的基础,它能够同时将消息推送给多个队列。

主题交换机(Topic Exchange):交换机会根据路由键进行模糊匹配,从而完成消息投送。

头交换机(Header Exchange):它不依赖特定路由键,而是将投送目标写在消息头,支持字典类型,配置更加灵活。

二、C++开发指南

官网提供了其它常见语言的开发向导,对于C++个人推荐使用AMQP-CPP这套库。另外还需要一套网络库支持,个人也推荐libevent。编译方法可以参考github上的说明。发送方式区别于传统的socket,你不应该将一条消息分多个部分发送。因此推荐使用对象序列化模型直接转换为字节数组,同样受到tcp/ip传输的制约,你应该选择高效的序列化工具来进行。个人推荐使用protobuf,同样作为一种跨平台的支持。

下面以一套RPC调用为例进行说明:

#include <iostream>
#include "event2/event.h"
#include "amqpcpp.h"
#include "amqpcpp/libevent.h"
#include "amqp_msg.pb.h"
#include <string> using namespace std;
using namespace amqp; int main()
{
event_base *base = event_base_new(); // 通过libevent启动实践循环
AMQP::LibEventHandler handler(base);
AMQP::TcpConnection connection(&handler, AMQP::Address("localhost", 5672, AMQP::Login("guest", "guest"), "/"));
AMQP::TcpChannel channel(&connection); // 创建一条通道
channel.setQos(1);
// 监听login.rpc队列
channel.consume("login.rpc").onReceived([&](const AMQP::Message &message, uint64_t deliveryTag, bool redelivered)
{
cout << "login.rpc" << endl;
Login login;
login.ParseFromArray(message.body(), message.bodySize());
Response resp; // 创建应答对象
resp.set_status(Response_RespType_OK);
resp.set_session_id("acd");
char data[1024] = {0};
int data_size = resp.ByteSizeLong();
resp.SerializeToArray(data, data_size);
AMQP::Envelope env(data, data_size);
env.setCorrelationID(message.correlationID()); // 获取应答ID
channel.publish("", message.replyTo(), env); // 发送给应答队列
channel.ack(deliveryTag); // 向MQ发送确认
}).onSuccess([&](const std::string &consumertag)
{ }).onError([](const char *message)
{
event_base_loopbreak(base); // 发送错误中断事件循环
cout << message << endl;
});
// 监听logout.rpc队列
channel.consume("logout.rpc")
.onReceived([&channel](const AMQP::Message &message, uint64_t deliveryTag, bool)
{
Logout logout;
logout.ParseFromArray(message.body(), message.bodySize());
Response resp;
resp.set_status(Response_RespType_OK);
char data[1024] = {0};
int data_size = resp.ByteSizeLong();
resp.SerializeToArray(data, data_size);
AMQP::Envelope env(data, data_size);
env.setCorrelationID(message.correlationID());
channel.publish("", message.replyTo(), env);
channel.ack(deliveryTag);
}).onError([](const char *message)
{
event_base_loopbreak(base);
cout << message << endl;
});
event_base_dispatch(base); // 事件循环
event_base_free(base); // 释放
return 0;
}

AMQP-CPP库直接主动连接,或者你也可以在继承相应的Handler自己完成网络连接。此外,Connection 和 Channel的创建也都支持回调函数。如:

channel.onError([&base](const char* message)
{
std::cout << "Channel error: " << message << std::endl;
event_base_loopbreak(base);
});
channel.declareQueue("queueName", AMQP::passive)
.onSuccess([&](const string& name, uint32_t, uint32_t)
{
cout << "Queue Name:" << name << endl;
});
channel.declareExchange("logs", AMQP::fanout)
.onSuccess([&]() {})

三、Spring AMQP开发指南

与Spring整合的技巧,官网有很详细的指导意见。这里只给出与上文C++配合的请求端如何发送以及等待应答的核心代码:

@GetMapping("login")
public String loginRpc() throws InvalidProtocolBufferException {
AmqpMsg.Login login = AmqpMsg.Login.newBuilder()
.addParams(AmqpMsg.PairParams.newBuilder().setKey("username").setValue("admin").build())
.addParams(AmqpMsg.PairParams.newBuilder().setKey("password").setValue("admin").build())
.build();
byte[] resp = (byte[]) template.convertSendAndReceive(directExchange.getName(), "login.rpc", login.toByteArray()); AmqpMsg.Response response = AmqpMsg.Response.parseFrom(resp);
if (response.getStatus() == AmqpMsg.Response.RespType.OK) {
String sessionID = response.getSessionId();
System.out.println("登录成功 SessionID=" + sessionID);
return "SUCCESS";
}
return "ERROR";
} @GetMapping("logout")
public String logoutRpc() throws InvalidProtocolBufferException {
AmqpMsg.Logout logout = AmqpMsg.Logout.newBuilder()
.setSessionId("123456").build();
byte[] resp = (byte[]) template.convertSendAndReceive(directExchange.getName(), "logout.rpc", logout.toByteArray());
AmqpMsg.Response response = AmqpMsg.Response.parseFrom(resp);
if(response.getStatus() == AmqpMsg.Response.RespType.OK) {
System.out.println("注销成功");
return "SUCCESS";
}
return "ERROR";
}
@Configuration
public class RPCRabbitConfig {
@Bean("simple")
public Queue simpleQueue() {
return new Queue("simple");
} @Bean("login.rpc")
public Queue loginRpcQueue() {
return new Queue("login.rpc");
} @Bean("logout.rpc")
public Queue logoutRpcQueue() {
return new Queue("logout.rpc");
} @Bean
public DirectExchange defaultExchange() {
return new DirectExchange("amq.direct");
} @Bean
public Binding loginRpcBinding(DirectExchange exchange, @Qualifier("login.rpc") Queue queue) {
return BindingBuilder.bind(queue).to(exchange).with("login.rpc");
} @Bean
public Binding logoutRpcBind(DirectExchange exchange, @Qualifier("logout.rpc") Queue queue) {
return BindingBuilder.bind(queue).to(exchange).with("logout.rpc");
}
}

后记:可能是由于工作上与架构的关系比较密切,目前在博客中提供的大多数解决方案都以跨平台应用为主。如果您对文章中介绍的知识点有任何的疑问也可以与我联系。

RabbitMQ协议基础及C++和Java混合开发的更多相关文章

  1. udp协议基础(转自疯狂java讲义)

    第17章  网络编程 17.4  基于UDP协议的网络编程 UDP协议是一种不可靠的网络协议,它在通信实例的两端各建立一个Socket,但这两个Socket之间并没有虚拟链路,这两个Socket只是发 ...

  2. RabbitMQ的基础介绍

    转自:http://blog.csdn.net/whycold/article/details/41119807 一.引言 你是否遇到过两个(多个)系统间需要通过定时任务来同步某些数据?你是否在为异构 ...

  3. Flutter基础系列之混合开发(二)

    1.混合开发的场景 1.1作为独立页面加入 这是以页面级作为独立的模块加入,而不是页面的某个元素. 原生页面可以打开Flutter页面 Flutter页面可以打开原生页面 1.2作为页面的一部分嵌入 ...

  4. Atitit onvif协议获取rtsp地址播放java语言 attilx总结

    Atitit onvif协议获取rtsp地址播放java语言 attilx总结 1.1. 获取rtsp地址的算法与流程1 1.2. Onvif摄像头的发现,ws的发现机制,使用xcf类库1 2. 调用 ...

  5. Java基础学习总结(70)——开发Java项目常用的工具汇总

    要想全面了解java开发工具,我们首先需要先了解一下java程序的开发过程,通过这个过程我们能够了解到java开发都需要用到那些工具. 首先我们先了解完整项目开发过程,如图所示: 从上图中我们能看到一 ...

  6. RabbitMQ (基础知识)

    AMQP简介 AMQP(Advanced Message Queue )即:高级消息队列协议:,是应用层协议的一个开放标准,为面向消息的中间件设计:高级消息队列协议使得遵从该规范的客户端应用和消息中间 ...

  7. 零基础如何系统学习Java Web

    零基础如何系统学习Java Web?   我来给你说一说 你要下决心,我要转行做开发,这样你才能学成. 你要会打字,我公司原来有一个程序员,打字都是两个手一指禅,身为程序员你一指禅怎么写出的代码,半个 ...

  8. android混合开发,webview的java与js互操作

    android原生应用,用webview加载应用中的网页,并且java代码与js代码可以互相操作. 这是混合开发的基石,最基本也最重要的东西,实验代码在这里. 概括说说—— java调js:调用web ...

  9. TCP/IP协议基础(转)

    转自 http://www.chinaunix.net 作者:Bernardus160  发表于:2003-12-03 17:33:15 TCP/IP协议基础 -------------------- ...

  10. Jvm基础(2)-Java内存模型

    Jvm基础(2)-Java内存模型 主内存和工作内存 Java内存模型包括主内存和工作内存两个部分:主内存用来存储线程之间的共享变量:而工作内存中存储每个线程的相关变量. 如下图所示: 需要注意的是: ...

随机推荐

  1. 缓存框架 Caffeine 的可视化探索与实践

    作者:vivo 互联网服务器团队-  Wang Zhi Caffeine 作为一个高性能的缓存框架而被大量使用.本文基于Caffeine已有的基础进行定制化开发实现可视化功能. 一.背景 Caffei ...

  2. android常用布局基础学习

    总结:可水平放置可垂直放置也可穿插使用,默认为水平 <!--我在第一次使用权重的时候忽视了本线性布局中的宽度与高度,如果要使用权重,请将线性布局的最初大小设置为match_parent,否则不会 ...

  3. M1安装Anaconda遇到的问题

    1. 安装时报错:"Anaconda3 is already installed in /opt/anaconda3. Use 'conda update anaconda3' to upd ...

  4. Known框架实战演练——进销存财务管理

    本文介绍如何实现进销存管理系统的财务对账模块,财务对账模块包括供应商对账和客户对账2个菜单页面.供应商和客户对账字段相同,因此可共用一个页面组件类. 项目代码:JxcLite 开源地址: https: ...

  5. 【MongoDB】Re05 分片集群(Win平台搭建)

    分片副本集1 (3实例)  主1 从1 裁1 分片副本集2 (3实例)  主1 从1 裁1 配置副本集(3实例) 主1 从2 路由(2配置) 用Windows平台搭建 配置目录设置: ├─config ...

  6. 【MybatisPlus】 Field '主键' doesn't have a default value

    使用MybatisPlus的 PoMapper执行Insert插入方法报错: 复原场景: 1.PO对象存在主键值(双主键) 2.表中数据为空 3.首次插入 这张表使用的是双主键,发现原因是因为PO设置 ...

  7. 【Vue】Re17 Router 第四部分(参数传递,守卫函数)

    一.案例搭建 新建Profile组件 组件写好内容后配置路由 { path : '/profile', component : () => import('../components/Profi ...

  8. 【Vue】10 Vue-Cli 项目创建

    简单的Demo案例并不需要Vue-Cli,因为一个页面之内可以总揽 但是真实的项目开发,考虑代码结构,目录结构,部署,热部署,单元测试... 代码量呈几何倍数增长,而且缺少轮子就写起来很痛苦 所以必须 ...

  9. 【产品兼容认证】WhaleStudio 成功兼容TiDB数据库软件

    平凯星辰和白鲸开源宣布成功完成产品兼容认证 北京,2023年12月27日 - 平凯星辰(北京)科技有限公司(以下简称平凯星辰)旗下的 TiDB 产品与白鲸开源的 WhaleStudio 已成功完成产品 ...

  10. MFC制作带界面的DLL库

    ## MFC如何创建一个带界面的DLL(动态链接库) 1.创建项目 打开VS,文件->新建->项目: 点击确定之后弹出来的界面,点击下一步->选择"使用共享MFC DLL的 ...