首先,我想说这是另一个问题:Similar but not the same
我的代码如下所示:
struct addrinfo hints, *res;
struct sockaddr* serveraddr;
memset(&hints, 0, sizeof(hints));
hints.ai_family = AF_INET;
hints.ai_socktype = SOCK_STREAM;
int res2 = getaddrinfo(ip, port, &hints, &res);
printf("getaddrinfo() res: %d, %d\n", res2, errno);
serveraddr = res->ai_addr;
//create new socket
int soc = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
printf("socket() res: %d, %d\n", socket, errno);
//set nonblocking mode
unsigned long on = 1;
res2 = ioctl(soc, FIONBIO, &on);
printf("ioctl() res: %d\n, %d", res2, errno);
res2 = connect(soc, serveraddr, sizeof(struct sockaddr));
printf("connect() res: %d, %d\n", res2, errno);
//check if socket is ready
fd_set wfds;
FD_ZERO(&wfds);
FD_SET(soc, &wfds);
struct timeval t;
t.tv_sec=15;
t.tv_usec=0;
res2 = select(soc + 1, 0, &wfds, 0, &t);
printf("select() res: %d, %d\n", res2, errno);`
我正在连接到存在的机器(IP地址存在,但是没有服务器正在侦听正在尝试连接的端口)。
选择始终返回1.为什么?根据男人,它应该超时并返回0。 在此之后,当我尝试向socket写入内容时,它返回-1 / ECONNREFUSED。
这是预期的行为吗?如果是,如何在select()之后检查我们是否已连接?
答案 0 :(得分:8)
我不知道你看到它应该超时的手册页在哪里。
如果没有防火墙丢弃数据包,连接将被很快拒绝(来自主机的一个数据包,一个数据包回复)。因此,一旦接收到重置,连接插座上的“事件”就会进入。这将唤醒select
,并且(至少)有一个活动套接字。
首次尝试读取或写入该套接字将返回基础连接错误。
答案 1 :(得分:1)
这是correct behavior,因为挂起的错误导致套接字变得既可读又可写。与任何其他系统调用一样,您需要明确检查错误情况。您可以在异常fdset中包含套接字(select()的第四个参数),并直接接收通知,或者稍后可以检查a number of ways中的错误。
小心写入套接字,因为可能发生SIGPIPE。通常,您应该设置一个SIGPIPE处理程序并与EPIPE同步处理错误。