Bind返回正在使用的地址,即使没有建立连接

bind returns address in use even if no connection is established

本文关键字:连接 建立 返回 Bind 地址      更新时间:2023-10-16

我有一个c++代码,我试图在套接字上建立连接。但我首先需要检查给定端口上是否已经存在连接,如果存在,我需要关闭连接。我有下面的代码,我的问题是,当检查端口是否已经连接时,它返回即使连接之前失败。

connected = false;
int sockfd;
void conn(int port) {
   struct sockaddr_in addr;
   addr.sin_family = AF_INET;
   addr.sin_port   = htons(port);
   .....
   int sockfd_t;
   if ( (sockfd_t = socket(PF_INET, SOCK_STREAM, 0)) < 0) {
       cout << "Error opening socket_test " << endl;
       return;
   }
   // check if address already in use
   if (bind(sockfd_t, (struct sockaddr *) &addr, sizeof(addr)) < 0) {
        if(errno == EADDRINUSE) {
        cout << "address in use: bind fail, port=" << port << endl;
    }
        // do something - close the connection if already connected
    }
    else {
        cout << "bind ok, port=" << port << endl;
    }
    close(sockfd_t);
    if ( (sockfd = socket(PF_INET, SOCK_STREAM, 0)) < 0) {
       cout << "Error opening socket " << endl;
       return;
    }
    if (connect(sockfd, (struct sockaddr *) &adresse, sizeof(adresse))) {
        cout << "Error connecting" << endl;
        close(sockfd);
        return;
    }
    connected = true;
}
int main() {
    int port=3590;
    while (!connected) {
       conn(port);
    }
    cout << "CONNECTED";
    // ..........
}

程序运行后输出如下:

bind ok, port=3590;
Error connecting
bind ok, port=3590;
Error connecting
address in use: bind fail, port=3590 //???
CONNECTED!

我不知道为什么在打印的第5行它显示"地址在使用:…"作为连接失败的前两次?

我想你对这些套接字操作的作用有误解。

但是我首先需要检查在给定端口上是否已经存在连接,如果存在,我需要关闭连接

  1. bind()给套接字一个本地地址,而不是检查您试图连接的远程地址是否可访问。

  2. connect()将套接字连接到远端地址

  3. 当连接一个套接字作为客户端(这是我认为你正在尝试做的),你不需要检查是否已经有一个连接,远程服务器可以处理多个传入的客户端连接到同一端口。

  4. 如果你在连接之前没有绑定,socket将被分配一个随机的本地端口。

所以,如果你是一个客户端,你做:

  1. socket()
  2. connect()

如果您是服务器,您可以:

  1. socket()
  2. bind()
  3. listen()

在您自己的问题中,当没有服务器侦听时,输出是有意义的,但随后服务器联机。

  1. 前两次,你绑定一个套接字,它是成功的,因为没有人使用它来监听,然后你连接失败,因为你只是绑定,并没有启动服务器(通过调用listen())。

  2. 然后同一主机上的真实服务器绑定该套接字并开始侦听,因此您不能再绑定该端口(它失败了),但是您可以连接,因为服务器正在侦听。