概述

最近项目消息队列服务选用了rabbitmq,server端用的C++开发的,于是需要开发rabbitmq的c++客户端,国际惯例先百度了一圈,然后github搜了一圈,竟然发现排名靠前的需要付费才能使用,尼玛这都拿出来骗钱(愤青了),于是产生了写个客户端给大家使用的念头。

我的应用场景是这样的生产者是用java写的(java别说了,一搜一大堆,改个服务器地址就能用),服务端作为消费者,老本行用的C++开发的,只需写个c++的rabbitmq客户端嵌入到我的server里就行,所有的消费者监听的一个队列,无需指定routing key和exchange,任何一个消费者消费到数据就行(我的服务端是个集群)。话不多说上代码。

实现

我的环境是Linux centos7.2

amqpclient.cpp源码如下:

#include <unistd.h>
#include <stdlib.h>
#include <stdio.h>
#include <string.h>
#include <stdint.h> #include "amqp_tcp_socket.h"
#include "amqp.h"
#include "amqp_framing.h"
#include "utils.h"
#include "platform_utils.h"
#include "amqpclient.h" /*
* 构造函数
* hostname,port,user,pwd,vhost
*/ AmqpClient::AmqpClient(const char* hostname, const int port, const char* user,
const char* pwd, const char* vhost, const char* queuename)
{
if((hostname != NULL) || (port != 0) || (user != NULL)
|| (pwd != NULL) || (vhost != NULL) || (queuename != NULL))
{
strcpy(this->m_hostname, hostname);
this->m_port = port;
strcpy(this->m_user, user);
strcpy(this->m_pwd, pwd);
strcpy(this->m_vhost, vhost);
strcpy(this->m_queuename, queuename);
}
} bool AmqpClient::ConnectRabbitmq()
{
amqp_socket_t *socket ;
amqp_rpc_reply_t reply ;
int status; conn = amqp_new_connection();
socket = amqp_tcp_socket_new(conn);
if(!socket)
{
printf("amqp new socket error\n");
return 0;
}
status = amqp_socket_open(socket, m_hostname, m_port);
if(status)
{
printf("amqp open socket error\n");
return 0;
}
reply = amqp_login(conn, m_vhost, 0, 131072, 0, AMQP_SASL_METHOD_PLAIN, m_user, m_pwd);
if (reply.reply_type == AMQP_RESPONSE_SERVER_EXCEPTION)
{
printf("amqp login error\n");
return 0;
}
amqp_channel_open(conn, 1);
reply = amqp_get_rpc_reply(conn);
if(reply.reply_type == AMQP_RESPONSE_SERVER_EXCEPTION)
{
printf("ConnectRabbitmq::amqp get rpc_reply error\n");
return 0;
}
} bool AmqpClient::StartConsumer()
{
amqp_rpc_reply_t reply ;
//自动回复ACK
amqp_basic_consume(conn, 1, amqp_cstring_bytes(m_queuename), amqp_empty_bytes, 0, 1, 0, amqp_empty_table);
reply = amqp_get_rpc_reply(conn);
if(reply.reply_type == AMQP_RESPONSE_SERVER_EXCEPTION)
{
printf("StartConsumer::amqp get rpc_reply error\n");
return false;
}
{
for (;;)
{
amqp_rpc_reply_t res;
amqp_envelope_t envelope; amqp_maybe_release_buffers(conn); res = amqp_consume_message(conn, &envelope, NULL, 0); if (AMQP_RESPONSE_NORMAL != res.reply_type) {
break;
} printf("Delivery %u, exchange %.*s routingkey %.*s\n",
(unsigned) envelope.delivery_tag,
(int) envelope.exchange.len, (char *) envelope.exchange.bytes,
(int) envelope.routing_key.len, (char *) envelope.routing_key.bytes); if (envelope.message.properties._flags & AMQP_BASIC_CONTENT_TYPE_FLAG) {
printf("Content-type: %.*s\n",
(int) envelope.message.properties.content_type.len,
(char *) envelope.message.properties.content_type.bytes);
}
printf("----\n"); amqp_dump(envelope.message.body.bytes, envelope.message.body.len); amqp_destroy_envelope(&envelope);
}
}
} void AmqpClient::CloseConnect()
{
amqp_rpc_reply_t reply ;
amqp_channel_close(conn, 1, AMQP_REPLY_SUCCESS);
amqp_connection_close(conn, AMQP_REPLY_SUCCESS);
amqp_destroy_connection(conn);
}

amqpclient.h源码如下:

#ifndef _AMQPCLIENT_H
#define _AMQPCLIENT_H #include "amqp.h" class AmqpClient
{
public:
AmqpClient(const char* hostname, const int port, const char* user, const char* pwd,
const char* vhost, const char* queuename);
bool ConnectRabbitmq();
bool StartConsumer();
void CloseConnect();
private:
amqp_connection_state_t conn = NULL;
char m_hostname[128];
int m_port;
char m_user[128];
char m_pwd[128];
char m_vhost[128];
char m_queuename[128];
}; #endif

main.cpp源码如下:

#include <stdio.h>
#include "amqpclient.h" int main()
{
const char* hostname = "192.168.12.20";
const int port = 5672;
const char* user = "woniu201";
const char* pwd = "woniu201";
const char* vhost = "/";
const char* queuename = "queuename001"; AmqpClient amqpClient(hostname, port, user, pwd, vhost, queuename);
amqpClient.ConnectRabbitmq();
printf("connect rabbitmq succ, is consuming!\n");
amqpClient.StartConsumer(); return 1;
}

Makefile文件:

EXE=rabbitmq_consumer
SUBDIR=src #CXXFLAGS:编译选项, LDFLAGS:链接选项
CXXFLAGS +=
LDFLAGS += -L/usr/local/lib -lrabbitmq CXX_SOURCES =$(foreach dir,$(SUBDIR), $(wildcard $(dir)/*.cpp))
CXX_OBJECTS=$(patsubst %.cpp, %.o, $(CXX_SOURCES))
DEP_FILES =$(patsubst %.o, %.d, $(CXX_OBJECTS)) $(EXE): $(CXX_OBJECTS)
g++ $(CXX_OBJECTS) -o $(EXE) $(LDFLAGS) %.o: %.cpp
g++ -c $(CXXFLAGS) -MMD $< -o $@ -include $(DEP_FILES) clean:
rm -rf $(CXX_OBJECTS) $(DEP_FILES) $(EXE) test:
echo $(CXX_OBJECTS) initlib:
ln -s lib/librabbitmq.so.4.2.0 /usr/local/lib/librabbitmq.so
ln -s lib/librabbitmq.so.4.2.0 /usr/lib64/librabbitmq.so.4

这里用到了librabbitmq.so.4动态库,make前先执行make initlib指令,会把所用的librabbitmq.so库放到系统环境中。

make完后会生产rabbitmq_consumer可执行文件,./rabbitmq_consumer执行。

测试结果如下:

如果到这里还有问题可以直接下载我的源码或者留言交流:http://download.csdn.net/detail/woniu211111/9911075

没有下载积分的,可以关注下面公众号,回复"107"获取源码

欢迎加群交流:C/C++开发交流

Rabbitmq C++客户端 Rabbitmq Client的更多相关文章

  1. RabbitMQ C#客户端自动重连

    重要参考文章来源:http://gigi.nullneuron.net/gigilabs/resilient-connections-with-rabbitmq-net-client/ 参考代码:ht ...

  2. C#基于RabbitMQ实现客户端之间消息通讯实战演练

    一.背景介绍和描述 MQ消息队列已经逐渐成为企业IT系统内部通信的核心手段.它具有低耦合.可靠投递.广播.流量控制.最终一致性等一系列功能,成为异步RPC的主要手段之一.何时需要消息队列?当你需要使用 ...

  3. RabbitMQ JAVA客户端调用例子

    1.安装erlang 下载地址:http://www.erlang.org/downloads 设置ERLANG环境变量 2.安装RabbitMQ 下载地址: http://www.rabbitmq. ...

  4. RabbitMQ GUI客户端工具(RabbitMQ Assistant)

    RabbitMQ GUI客户端工具(RabbitMQ Assistant) 平时用控制台或者网页进行管理不免有点不方便,尤其在读取消息的时候不支持过滤和批量发送消息,在此推荐一个漂亮的GUI客户端工具 ...

  5. RabbitMQ交换机、RabbitMQ整合springCloud

    目标 1.交换机 2.RabbitMQ整合springCloud 交换机 蓝色区域===生产者 红色区域===Server:又称Broker,接受客户端的连接,实现AMQP实体服务 绿色区域===消费 ...

  6. .Net RabbitMQ实战指南——RabbitMQ相关概念介绍

    什么是消息中间件 消息(Message)是指在应用间传送的数据.消息可以非常简单,比如只包含文本字符串.JSON等,也可以很复杂,比如内嵌对象. 消息队列中间件(Message Queue Middl ...

  7. RabbitMQ入门到进阶(Spring整合RabbitMQ&SpringBoot整合RabbitMQ)

    1.MQ简介 MQ 全称为 Message Queue,是在消息的传输过程中保存消息的容器.多用于分布式系统 之间进行通信. 2.为什么要用 MQ 1.流量消峰 没使用MQ 使用了MQ 2.应用解耦 ...

  8. .NET文件并发与RabbitMQ(初探RabbitMQ)

    本文版权归博客园和作者吴双本人共同所有.欢迎转载,转载和爬虫请注明原文地址:http://www.cnblogs.com/tdws/p/5860668.html 想必MQ这两个字母对于各位前辈们和老司 ...

  9. 基于 IdentityServer3 实现 OAuth 2.0 授权服务【客户端模式(Client Credentials Grant)】

    github:https://github.com/IdentityServer/IdentityServer3/ documentation:https://identityserver.githu ...

随机推荐

  1. swoole入门到实战打造高性能赛事直播平台☆

    ​ 第1章 课程介绍 本章主要是介绍了swoole的一些特性,以及使用场景,并且分享了swoole在其他公司的一些案例,最后重点讲解了swoole学习的一些准备工作. 第2章 PHP 7 源码安装 本 ...

  2. docker安装ubuntu以后无ifconfig命令解决办法

    解决: 1.apt-get  update 2.apt install net-tools   #ifcongig 3.apt install iputils-ping     # ping

  3. Eclipse简明使用教程(java集成开发环境)

    说明:我使用的是绿色便携版的Eclipse,免安装,解压就能用.链接:https://download.csdn.net/download/loongstyle/10700321 1_Eclipse中 ...

  4. nodejs做中间层,转发请求

    简述node中间层的优势 node中间层,可以解决前端的跨域问题,因为服务器端的请求是不涉及跨域的,跨域是浏览器的同源策略导致的,关于跨域可以查看跨域复习使用node坐中间层,方便前后端分离,后端只需 ...

  5. cin中的注意事项

    猎豹网校C++ Primer学习笔记 - ostartech - 博客园 https://www.cnblogs.com/wxl845235800/p/11396323.html 在VS中使用cin& ...

  6. Spring Boot属性配置&自定义属性配置

    一.修改默认配置 例1.spring boot 开发web应用的时候,默认tomcat的启动端口为8080,如果需要修改默认的端口,则需要在application.properties 添加以下记录: ...

  7. setShadpwLayer实现阴影效果

    package com.loaderman.customviewdemo; import android.content.Context; import android.graphics.*; imp ...

  8. 自定义控件之canvas变换和裁剪

    1.平移 //构造两个画笔,一个红色,一个绿色 Paint paint_green = generatePaint(Color.GREEN, Paint.Style.STROKE, 3); Paint ...

  9. javascript——语法 && 结构

    原文链接:Understanding Syntax and Code Structure

  10. React Native 中的 Flex Box 的用法(水平布局、垂直布局、水平居中、垂直居中、居中布局)

     版权声明:本文仅供个人学习. CSS 中 Flex-Box 语法链接 http://www.ruanyifeng.com/blog/2015/07/flex-grammar.html Flex 是 ...