以前遇到过,这次复习到了故做个记录。这主要是因为上一个绑定该端口的套接字还未真正解绑导致的,原本该端口已经被一个套接字所绑定, 但由于该套接字建立的TCP连接还处于TIME_WAIT状态, 导致端口和套接字之间的绑定还未解除, 故不允许其它套接字绑定该端口。一般发生于一个TCP服务端程序挂掉后,试图立刻重启,并重写绑定原本端口的时候。
解决方法:设置允许端口重用,对绑定端口的套接字设置SO_REUSEADDR选项
int opt=1;
// 设置允许端口重用
if (setsockopt(listenFd, SOL_SOCKET, SO_REUSEADDR, &opt, sizeof(opt)) < 0)
{
perror("setsockopt");
}
setsockopt
函数是一个用于设置套接字选项的系统调用。它允许程序员控制套接字行为,比如改变缓冲区大小、设置超时时间或者修改其他底层网络行为。
这个函数的原型如下:
int setsockopt(int sockfd, int level, int optname, const void *optval, socklen_t optlen);
sockfd
:套接字文件描述符。level
:选项所在的协议层。SOL_SOCKET
是一个宏,表示正在设置的选项位于套接字层。optname
:需要访问的选项名。例如,SO_REUSEADDR
是一个宏,用于设置地址重用。optval
:指向包含新选项值的缓冲区的指针。optlen
:optval
缓冲区的大小。
SOL_SOCKET
是一个宏,用于指定选项代码的级别,表示这些选项是与套接字相关的,而不是与特定协议相关的。这意味着SOL_SOCKET
级别的选项适用于所有类型的套接字。
SO_REUSEADDR
是一个宏,用于设置地址重用选项。当套接字使用这个选项时,即使前一个绑定到该端口的套接字的连接仍处于TIME_WAIT
状态,也允许另一个套接字绑定到同一个端口。这在开发网络服务时非常有用,因为它允许服务快速重启,而不需要等待端口从TIME_WAIT
状态释放。