11

我有一个非常简单的问题。我想测试一个特定的端口当前是否正在使用中。为此,我想将 TCP 套接字绑定到端口,如果连接被拒绝,则表示该端口正在使用中,如果不是,则表示该端口是空闲的。

有人可以告诉我如何用 C 编写 TCP 套接字代码吗?我在一个solaris平台上。

我知道它非常基本。但我感谢你的帮助。提前致谢。

4

4 回答 4

12

如果发生错误,对绑定函数的调用将返回 -1。这包括地址已被使用的情况。

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

#define PORT 12345

int main()
{
  struct sockaddr_in addr;
  int fd;

  fd = socket(AF_INET, SOCK_STREAM, 0);
  if(fd == -1)
  {
      printf("Error opening socket\n");
      return -1;
  }

  addr.sin_port = htons(PORT);
  addr.sin_addr.s_addr = 0;
  addr.sin_addr.s_addr = INADDR_ANY;
  addr.sin_family = AF_INET;

  if(bind(fd, (struct sockaddr *)&addr,sizeof(struct sockaddr_in) ) == -1)
  {
      printf("Error binding socket\n");
      return -1;
  }

  printf("Successfully bound to port %u\n", PORT);
}
于 2008-10-15T10:13:32.300 回答
8

这稍微取决于您要测试的内容。

bind()joelc 建议的方式使用将告诉您该端口是否在您机器上的任何接口上打开。虽然要彻底,但您不仅应该检查 的返回值bind(),还应该检查errno == EADDRINUSE.

IE。(修改joelc的代码)


if(bind(socket, (struct sockaddr *)&sin,sizeof(struct sockaddr_in) ) == -1)
{
    if( errno == EADDRINUSE )
    {
        // handle port already open case
    }
    else
    {
        // handle other errors
    }
}

通过更改行中使用的地址:例如。


    sin.sin_addr.s_addr = inet_addr("192.168.1.1");

...您可以测试特定接口上的端口是否可用。

请注意,这不是对端口状态的完美测试。如果另一个进程打开了端口并在它正常关闭端口之前终止(即在调用close()套接字之前),那么您通常会遇到相同的EADDRINUSE错误。(取决于是否在套接字上设置了 SO_REUSEADDR 选项)

(旁注:除非您的测试应用程序以足够的权限运行,否则您将无法 bind() 到 1024 以下的任何端口)

正如 Anonymous 建议的那样,您也可以查看netstat. 这将为您提供所有相同的信息,您可以通过bind()更快地重复调用并且没有任何副作用(例如它不必实际绑定到端口,这会使它们无法用于任何其他进程) . 只需调用netstat -a --numeric-ports -t和分析输出即可为您提供所需的一切。

不过,对 moogs 建议的评论 - 在每个端口上调用 telnet 只会告诉您套接字是否正在侦听该端口 - 而不是它是否实际打开。

于 2008-10-15T11:17:02.263 回答
-1

您是否只想测试特定端口当前是否正在使用?(并且真的不需要制作程序)。如果是这样,您可以使用 telnet:

远程登录主机端口

如果连接失败,则说明未使用。如果它连接并等待您的输入,则它正在使用中:)

于 2008-10-15T10:14:01.423 回答
-2

您可能想查看 netstat 的源代码。我相信 Solaris 中也有一个 netstat。

于 2008-10-15T10:06:27.367 回答