2

原始帖子: 我正在使用 libevent 和 zmq 编写 C 编程服务。使用 PUSH-PULL 模式将消息从 python 代码推送到 C 服务。

从 zmq 套接字接收到的 fd:

void *receiver = zmq_socket (base.zmq_ctx, ZMQ_PULL);
zmq_connect (receiver, "tcp://localhost:5557");
int fd=0;
size_t fd_len = sizeof(fd);
zmq_getsockopt (receiver, ZMQ_FD, &fd, &fd_len);

使用 Libevent,向 fd 注册事件以进行持久读取

struct event *read_data_on_zmq =event_new(base.evbase, fd, EV_READ | EV_PERSIST , read_data_on_zmq_cb,receiver);
event_add(read_data_on_zmq,NULL);
event_base_dispatch(base.evbase);

在回调方法上,我正在做一个非阻塞接收

void read_data_on_zmq_cb(evutil_socket_t fd, short what, void *arg)
{
    char *msg = calloc(1024,sizeof(char));
    int size = zmq_recv (receiver, msg, 255, ZMQ_DONTWAIT);
    if (size != -1) 
    {
        puts ("is size is not -1");
        printf("msg = %s\n",msg); 
    }
}

在 python 代码中,我不断地向套接字发送消息。

import zmq
import time

c=zmq.Context()
s=c.socket(zmq.PUSH)
s.bind('tcp://127.0.0.1:5557')
while(True):
    s.send("abc")
    time.sleep(2)

问题是我只能收到一次消息,之后事件回调永远不会被击中。如果我在 zmq_recv 之后的 read_data_on_zmq_cb 中执行 zmq_connect,那么它工作正常,但我想这是多余的,而不是正确的方法。这里有什么问题?

EDIT1: 除了在执行 zmq_recv() 之后检查 ZMQ_EVENTS 之外,您还需要获取所有消息,因为 zmq 是 EDGE 触发的。关于 EDGE 触发通知的一个很好的解释在这里http://funcptr.net/2012/09/10/zeromq---edge-triggered-notification/ 所以最终我的事件回调看起来像

void read_data_on_zmq_cb(evutil_socket_t fd, short what, void *arg)
{
    unsigned int     zmq_events;
    size_t           zmq_events_size  = sizeof(zmq_events);
    char *msg=NULL;

    zmq_getsockopt (receiver, ZMQ_EVENTS, &zmq_events, &zmq_events_size);

    while(zmq_events & ZMQ_POLLIN)
    {
        msg = calloc(1024,sizeof(char));
        int size = zmq_recv (receiver, msg, 255, ZMQ_DONTWAIT);
        if (size != -1) {
        #ifdef DEBUG    
            printf("msg = %s\n",msg);
        #endif
            //return msg;
        }
        zmq_getsockopt (receiver, ZMQ_EVENTS, &zmq_events, &zmq_events_size);
    }
}
4

1 回答 1

1

请阅读我的 EDIT1 以获得完整的代码答案。问题是:在执行 zmq_recv() 之后我没有重新检查 ZMQ_EVENTS,因为当时套接字的状态发生了变化。

所以打电话

zmq_getsockopt (receiver, ZMQ_EVENTS, &fd, &fd_size);

在 zmq_recv() 解决了我的问题之后。

于 2013-05-08T05:04:28.557 回答