该问题通常是由于代码中的连接关闭操作引起的。需要确保在调用连接函数后不会关闭本地套接字,直到连接关闭或超时,否则会中断连接调用。以下是一个示例,展示了一个可能会导致此问题的代码片段:
int fd = socket(AF_INET, SOCK_STREAM, 0);
// ...
connect(fd, (sockaddr*)&their_addr, sizeof(their_addr));
// ...
close(fd); // 非法的连接关闭操作,应该在连接关闭或超时后才能执行
为了解决该问题,可以在连接操作后仍然保留套接字文件描述符,并确保在连接关闭或超时后再关闭它。示例如下:
int fd = socket(AF_INET, SOCK_STREAM, 0);
// ...
connect(fd, (sockaddr*)&their_addr, sizeof(their_addr));
// ...
// 等待连接关闭或超时
fd_set rset, wset;
FD_ZERO(&rset);
FD_ZERO(&wset);
FD_SET(fd, &rset);
FD_SET(fd, &wset);
struct timeval tval;
tval.tv_sec = timeout_in_secs;
tval.tv_usec = 0;
int n = select(fd + 1, &rset, &wset, NULL, &tval);
if (n == 0) {
// 超时
} else if (n == -1) {
// 出错
} else {
if (FD_ISSET(fd, &rset) || FD_ISSET(fd, &wset)) {
// 连接已关闭
} else {
// 未知情况
}
}
close(fd); // 正确的连接关闭操作,应该在连接关闭或超时后才能执行