12

操作系统是Linux。我有一个可以实时更改其端口的服务器进程。但是我想提前知道绑定之前端口是否空闲。

场景:服务器绑定 localhost:5000 并在 localhost:6000 接收到绑定请求。服务器必须检查端口是否空闲。此问题寻求提供检查端口是否空闲的例程的答案。

作为记录,我正在使用代码片段编辑我的问题,该代码片段检查端口是否可以免费使用。这并不意味着它会被使用。下面的代码回答了“如果端口现在可用”的问题,它不使用它。打开一个套接字,检查 bind 是否返回 EADDRINUSE 并关闭套接字。

#include <iostream>
#include <sys/socket.h>
#include <stdlib.h>
#include <string.h>
#include <unistd.h>
#include <arpa/inet.h>
#include <errno.h>

int main(int argc, char **argv) {

    struct sockaddr_in serv_addr;
    if( argc < 2 )
        return 0;
    int port = atoi(argv[1]);

    int sockfd = socket(AF_INET, SOCK_STREAM, 0);
    if( sockfd < 0 ) {
        printf("socket error\n");
        return 0;
    } else {
        printf("Opened fd %d\n", sockfd);
    }

     bzero((char *) &serv_addr, sizeof(serv_addr));
     serv_addr.sin_family = AF_INET;
     serv_addr.sin_addr.s_addr = INADDR_ANY;
     serv_addr.sin_port = htons(port);
     if (bind(sockfd, (struct sockaddr *) &serv_addr, sizeof(serv_addr)) < 0) {

        if( errno == EADDRINUSE )
        {
            printf("the port is not available. already to other process\n");
        } else {
            printf("could not bind to process (%d) %s\n", errno, strerror(errno));
        }
    }

    if (close (sockfd) < 0 ) {
        printf("did not close fd: %s\n", strerror(errno));
        return errno;
    }

    return 0;


}

以下是一些示例运行(部分输出)

[bash{1051}{51}]:[~/some_sources/checkbind]::./a.out 41067
the port is not available. already to other process
[bash{1052}{52}]:[~/some_sources/checkbind]::./a.out 22
could not bind to process (13) Permission denied
[bash{1053}{53}]:[~/some_sources/checkbind]::./a.out 22000
Opened fd 3
4

3 回答 3

12

这是一个明显的竞争条件,因为您系统上的其他进程可能会并行绑定到端口。因此,您找到的任何解决方案都将是不完美的,您仍然需要根据“尝试bind(),如果失败选择一个新的端口号并重试”的方法来编写它。

于 2012-04-24T09:09:50.883 回答
8

我自己为此苦苦挣扎,并稍微修改了您的代码。

解决方案是设置serv_addr.sin_port = 0(自动分配端口)。

注意bind()getsockname()行中有从sockaddr_in到的不干净的演员表sockaddr。我已经看到它在很多地方做过,我正在寻找更安全的解决方案。

#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>

// ... snip ...

int sock = socket(AF_INET, SOCK_STREAM, 0);
if(sock < 0) {
    printf("socket error\n");
    return;
}
printf("Opened %d\n", sock);

struct sockaddr_in serv_addr;
bzero((char *) &serv_addr, sizeof(serv_addr));
serv_addr.sin_family = AF_INET;
serv_addr.sin_addr.s_addr = INADDR_ANY;
serv_addr.sin_port = 0;
if (bind(sock, (struct sockaddr *) &serv_addr, sizeof(serv_addr)) < 0) {
    if(errno == EADDRINUSE) {
        printf("the port is not available. already to other process\n");
        return;
    } else {
        printf("could not bind to process (%d) %s\n", errno, strerror(errno));
        return;
    }
}

socklen_t len = sizeof(serv_addr);
if (getsockname(sock, (struct sockaddr *)&serv_addr, &len) == -1) {
    perror("getsockname");
    return;
}

printf("port number %d\n", ntohs(serv_addr.sin_port));


if (close (sock) < 0 ) {
    printf("did not close: %s\n", strerror(errno));
    return;
}

程序输出

Opened 4
port number 59081
于 2013-12-31T01:12:06.817 回答
7

如果您的服务器被告知要使用哪个端口,就bind()这样吧。严重地。

当然,您可以解析/proc/net/tcp并查看端口是否正在使用中。但是然后呢?你现在仍然需要打电话bind(),因为你知道你的端口是空闲的,它会告诉你端口是否空闲,所以没有必要卑躬屈膝地/proc/net/tcp做所有这些(慢!)字符串生成和解析和额外的内核会通过不是很好优化的(阅读:与 相比超级慢bind())诊断路径,只是为了在您完成解析之前获取可能已经过时的信息。所以只要打电话bind(),开心就好。

于 2012-04-25T07:25:10.290 回答