一、检查步骤
- 使用socket函数创建socket_fd套接字。
- 使用sockaddr_in结构体配置协议和端口号。
- 使用bind函数尝试与端口进行绑定,成功返回0表示未被占用,失败返回-1表示已被占用。
二、步骤详解
2.1 socket函数
socket 函数是用于创建套接字的函数,其参数和返回值如下:
int socket(int domain, int type, int protocol);
输入参数
- domain(地址家族):指定套接字的地址家族,它表示了网络层协议的类型,通常是下列之一:
- AF_INET:IPv4 地址家族。
- AF_INET6:IPv6 地址家族。
- type(套接字类型):指定套接字的类型,它表示了传输层协议的类型,通常是下列之一:
- SOCK_STREAM:流套接字,用于 TCP 协议
- SOCK_DGRAM:数据报套接字,用于 UDP 协议。
- protocol(协议类型):通常设置为0,表示自动选择与套接字类型相匹配的默认协议。在大多数情况下,你不需要指定协议,因为操作系统会根据地址家族和套接字类型自动选择合适的协议。
返回值
- 如果 socket 函数成功返回值为socket_fd套接字描述符。
- 如果 socket 函数失败,它将返回-1,也就是INVALID_SOCKET。
2.2 配置协议和端口号
sockaddr_in结构体在winsock2.h(windows)或sys/socket.h(linux)头文件下。其结构如下:
// sin为socket in的缩写
struct sockaddr_in {
short sin_family; // 地址家族,通常为 AF_INET
unsigned short sin_port; // 端口号,以网络字节序表示
struct in_addr sin_addr; // IP地址
char sin_zero[8]; // 预留字段,通常置零
};
具体配置代码如下:
sockaddr_in serverAddress;
// 将serverAddress内存全部写0
ZeroMemory(&serverAddress,sizeof(serverAddress));
// AF_INET为ipv4,AF_INET6为ipv6
serverAddress.sin_family = AF_INET;
// htons是host to network short的缩写,表示将主机序以short类型转换为的网络序,并以二进制形式存储在sin_port里
serverAddress.sin_port = htons(port);
// INADDR_ANY表示支持任意地址
serverAddress.sin_addr.s_addr = INADDR_ANY;
2.3 bind函数
bind 函数用于将一个套接字与一个本地地址(通常是IP地址和端口号)绑定在一起,以便监听该地址上的连接或接收数据。bind 函数的参数和返回值如下:
int bind(int sockfd, const struct sockaddr *addr, socklen_t addrlen);
输入参数
- sockfd:是要绑定的套接字的文件描述符(在Unix/Linux环境中)或套接字句柄(在Windows环境中)。
- addr:是一个指向 struct sockaddr 类型的指针,用于指定要绑定的本地地址信息。通常需要将一个sockaddr_in类型的指针转换为sockaddr类型的指针。
- addrlen:是 addr 结构的长度,通常可以使用 sizeof 运算符获取。
返回值
- 如果 bind 函数成功绑定套接字到指定的地址,它将返回0。
- 如果 bind 函数失败,它将返回-1,表示绑定失败。失败的原因可能是指定的地址已经被占用或其他错误。
三、CODE
其中port需要修改为想要检测的端口号,也可以将代码改写为根据argv参数检测。
#include <iostream>
#include <winsock2.h> // Windows套接字编程头文件
// #include <sys/socket.h> // Unix/Linux套接字编程头文件
#include <Windows.h>
int main() {
// WSAStartup用于初始化 Winsock 库,它在进行套接字编程之前需要调用。
// MAKEWORD(2,2)表示请求2.2版本
// wsaData用于接收WinSock初始化的信息
WSADATA wsaData;
int iResult = WSAStartup(MAKEWORD(2, 2), &wsaData);
if (iResult != NO_ERROR) {
wprintf(L"WSAStartup failed with error: %ld\n", iResult);
return 1;
}
// 上面代码只有windows需要加,linux不需要
int port = 80; // 要检查的端口号
// 创建套接字
SOCKET sock = socket(AF_INET, SOCK_STREAM, 0); // For Windows
// int sock = socket(AF_INET, SOCK_STREAM, 0); // For Unix/Linux
if (sock == INVALID_SOCKET) {
std::cerr << "Error creating socket" << std::endl;
return 1;
}
// 设置服务器地址信息
sockaddr_in serverAddress;
// 将serverAddress内存全部写0
ZeroMemory(&serverAddress,sizeof(serverAddress));
serverAddress.sin_family = AF_INET;
serverAddress.sin_port = htons(port);
serverAddress.sin_addr.s_addr = INADDR_ANY;
// 尝试绑定套接字到指定端口
int result = bind(sock, (struct sockaddr*)&serverAddress, sizeof(serverAddress));
if (result == 0) {
std::cout << "Port " << port << " is available" << std::endl;
} else {
std::cerr << "Port " << port << " is already in use" << std::endl;
}
// 关闭套接字
closesocket(sock); // For Windows
// close(sock); // For Unix/Linux
return 0;
}
四、运行结果
- 使用以下命令行编译cpp代码,其中checkPort.cpp为源文件,checkPort为输出可执行文件,-lws2_32用于链接Windows的Winsock2库。
g++ .\checkPort.cpp -o checkPort -lws2_32
- 以下命令行运行可执行文件
.\checkPort.exe
- 如果端口未被占用,则会输出以下内容。
- 如果端口已被占用,则会输出以下内容。
- 如果windows下输出以下内容,一般来说是因为WSAStartup没有配置。