连接后非阻塞套接字选择返回1

时间:2011-12-07 15:26:28

标签: c linux sockets nonblocking select-function

首先,我想说这是另一个问题: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()之后检查我们是否已连接?

2 个答案:

答案 0 :(得分:8)

我不知道你看到它应该超时的手册页在哪里。

如果没有防火墙丢弃数据包,连接将被很快拒绝(来自主机的一个数据包,一个数据包回复)。因此,一旦接收到重置,连接插座上的“事件”就会进入。这将唤醒select,并且(至少)有一个活动套接字。

首次尝试读取或写入该套接字将返回基础连接错误。

答案 1 :(得分:1)

这是correct behavior,因为挂起的错误导致套接字变得既可读又可写。与任何其他系统调用一样,您需要明确检查错误情况。您可以在异常fdset中包含套接字(select()的第四个参数),并直接接收通知,或者稍后可以检查a number of ways中的错误。

小心写入套接字,因为可能发生SIGPIPE。通常,您应该设置一个SIGPIPE处理程序并与EPIPE同步处理错误。