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

C 非阻塞套接字选择在连接后返回1,c,linux,sockets,nonblocking,select-function,C,Linux,Sockets,Nonblocking,Select Function,首先,我想说的是,这是另一个问题,而不是这个问题: 我的代码如下所示: 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);

首先,我想说的是,这是另一个问题,而不是这个问题:

我的代码如下所示:

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。为什么?根据man的说法,它应该超时并返回0。 在此之后,当我尝试向套接字写入内容时,它返回-1/econnreference


这是预期的行为吗?如果是,在select()之后如何检查我们是否已连接?

我不知道在手册页的何处您看到它应该超时

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

第一次尝试读取或写入该套接字将返回底层连接错误。

这是因为挂起的错误会导致套接字既可读又可写。与任何其他系统调用一样,您需要明确检查错误条件。您可以将套接字包括在异常fdset(select()的第四个参数)中,并直接获取通知,或者稍后检查中的错误


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

?对如何检查?这个问题的答案很好。