ZeroMQ:重新绑定套接字时使用地址错误

ZeroMQ: Address in use error when re-binding socket

本文关键字:套接字 错误 地址 绑定 新绑定 ZeroMQ      更新时间:2023-10-16

将ZeroMQ套接字绑定到端点并关闭套接字后,将另一个套接字绑定到同一端点需要多次尝试。之前对zmq_bind的调用都失败了,错误是"地址在使用中"(EADDRINUSE)。

以下代码演示了这个问题:
#include <cassert>
#include <iostream>
#include "zmq.h"
int main() {
    void *ctx = zmq_ctx_new();
    assert( ctx );
    void *skt;
    skt = zmq_socket( ctx, ZMQ_REP );
    assert( skt );
    assert( zmq_bind( skt, "tcp://*:5555" ) == 0 );
    assert( zmq_close( skt ) == 0 );
    skt = zmq_socket( ctx, ZMQ_REP );
    assert( skt );
    int fail = 0;
    while ( zmq_bind( skt, "tcp://*:5555" ) ) { ++fail; }
    std::cout << fail << std::endl;
}

我在Windows XP SP3上使用ZeroMQ 4.0.3,编译器是VS 2008。libzmq.dll已使用提供的Visual Studio解决方案构建。

在做"调试"构建(上面的代码和libzmq.dll)和0使用"发布"构建时,在这里打印1。奇怪的是,当使用混合构建配置运行上面的代码时(使用发布库进行调试),fail的计数最多为6。

Pieter Hintjens在邮件列表中给了我提示:

调用zmq_close启动套接字关闭。这是在ZeroMQ启动的一个特殊的"死神"线程中完成的,以异步和非阻塞方式调用zmq_close。请参阅ZeroMQ架构白皮书中的"The reaper thread"。

上面的代码没有等待线程执行实际工作,因此端点不会立即可用。

当TCP套接字关闭时,它进入称为TIME_WAIT的状态。这意味着当套接字处于该状态时,它并没有真正关闭,这反过来又意味着套接字使用的地址在离开该状态之前是不可用的。

如果你连续运行你的程序两次,当你尝试第二次运行时,套接字将从第一次运行开始处于TIME_WAIT状态,你会得到一个像这样的错误。

您可能想了解更多关于TCP的信息,特别是关于它的操作和状态。