在linux下,使用AMQP-CPP开发rabbitmq c++

首先介绍一下背景以及我为什么选择AMQP来开发

1. rabbitmq作为我们的日志中转

2. 生产者只能为单进程(项目要求),且进程还需要处理其他事情,所以发送者不能够阻塞进程

3. 必须实现心跳功能(我们是与代理服务器相连,如果没有心跳且没传输数据时会被代理关闭连接,除非设置代理不关闭连接,比如nginx在十分钟没有数据的情况下就会端开连接,可以手动设置nginx的保活时间时间长一点来解决,但是治标不治本,所以最好实现心跳)

由于上述三点,我比较了SimpleAmqpClient 与AMQP-CPP,给出自己的观点:

1. SimpleAmqpClient与AMQP-CPP都是由c++封装的rabbitmq

2. SimpleAmqpClient不支持异步,发送会阻塞进程。AMQP-CPP采用回调,异步完成发送

3. 我没有找到SimpleAmqpClient的心跳发送函数,只找到设置,只能接收到服务器给的心跳,无法响应,所以会被断开(可能在其他地方,我没找到)。AMQP-CPP在回调onNegotiate中与服务器协商间隔,在onHeartbeat接收服务器心跳并手动调用heartbeat发送心跳。

4. SimpleAmqpClient的登录不能异步,这个在运行过程中如果重新登录也会阻塞住进程发的运行,毕竟我们只有一个进程。

5. SimpleAmqpClient的只能编译成动态库,如果要使用静态库需要改源码。AMQP-CPP可编译静态库与动态库。

注:SimpleAmqpClient可以实现非阻塞的登录,需要修改代码。

基于上述问题与特性,我选择AMQP-CPP来开发。

1. 在github上clone下来,地址:https://github.com/CopernicaMarketingSoftware/AMQP-CPP

2. 编译:

        mkdir build
        cd build
        cmake .. [-DAMQP-CPP_BUILD_SHARED] [-DAMQP-CPP_LINUX_TCP]
        cmake --build . --target install

        eg:cmake ..  -DAMQP-CPP_BUILD_SHARED=OFF -DAMQP-CPP_LINUX_TCP=ON

3. 使用AMQP-CPP

        AMQP-CPP可以自己实现网络部分,也可以直接用libevent,我选择直接用libevent

        首先:继承LibEventHandler,并实现自己的函数:

        

 class LibEventHandlerMyError : public AMQP::LibEventHandler                                   
 {
 public:
     LibEventHandlerMyError(struct event_base* evbase);
     
     ~LibEventHandlerMyError();
 
     void onError(AMQP::TcpConnection *connection, const char *message) override;
 
     virtual void onHeartbeat(AMQP::TcpConnection *connection);                                
 
     virtual void onReady(AMQP::TcpConnection *connection);
 
     virtual void onConnected(AMQP::TcpConnection *connection);
 
     virtual uint16_t onNegotiate(AMQP::TcpConnection *connection, uint16_t interval);
 public:                           
 
 private: 
    
     struct event_base* evbase_ {nullptr};
};

创建与rabbitmq的连接有三个类比较重要(我用的):AMQP::TcpConnection,AMQP::TcpChannel LibEventHandlerMyError(继承LibEventHandler,以使用网络)

1. auto event_base = event_base_new();

2.  handle =  new LibEventHandlerMyError(event_base, login);

3. tcp_connect_ = new AMQP::TcpConnection (this, AMQP::Address(host_.c_str(), port_,
            AMQP::Login(user_.c_str(), password_.c_str()), vhost_.c_str()));

4. tcp_channel_ = new AMQP::TcpChannel(tcp_connect_);

handle为LibEventHandlerMyError实例,login为我自己的登录结构体,创建TcpConnection时,我用的自己的LibEventHandlerMyError,所以用的this.

tcp_channel_就是用来操作的对象,发送,接收等

生产者过程:

 tcp_channel_->startTransaction();
 AMQP::Envelope env(msg, strlen(msg));
 env.setDeliveryMode(2);//这一步是设置属性,还有其他很多属性,具体的需要看头文件,我这里是设置为持久化信息

 tcp_channel_->publish(exchange, route_key, env); 
 tcp_channel_->commitTransaction()
   .onSuccess([]()mutable {
       std::cout<<"success"<<std::endl;
     })
   .onError([](const char *err_msg)mutable {
        std::cout<<"send msg fail:"<< err_msg<<std::endl;
     });

注意:写到此并不能将消息发送出去,因为我是单进程,如果你是多线程,你可以开一个线程去执行event_base_dispatch(evbase_)也就是 event_base_loop(evbase_, 0);这里将网络事件进行轮询监听,并调用相应的回调,记住:这个函数是死循环,也就是无法主动退出,必须手动调用event_base_loopbreak,但是我是单进程,所以我是手动调用event_base_loop(evbase_, 2),EVLOOP_NONBLOCK ,来触发监听。至此,生产者发送完毕。

消费者大同小异了
        

 tcp_channel_->consume(queue_name)
     .onReceived([](const AMQP::Message &msg, uint64_t tag, bool redelivered){
             .....todo
             channel->ack(tag);
             })  
 .onError([](const char *err_msg){
         printf("consume fail:%s", err_msg);
         });
             

over

  • 0
    点赞
  • 5
    收藏
    觉得还不错? 一键收藏
  • 1
    评论
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值