gpt4 book ai didi

c - 零超时的非阻塞 connect() 和 select() 返回 0

转载 作者:行者123 更新时间:2023-11-30 15:21:15 26 4
gpt4 key购买 nike

我正在编写一个单线程应用程序,它可以循环处理多个连接的 TCP 套接字。这是启动连接的部分:

// Set up connecting socket
wire[wi].skt=socket(AF_INET,SOCK_STREAM | SOCK_NONBLOCK,IPPROTO_TCP);
if (wire[wi].skt==-1) return TEC_SOCK_ERR;
wire[wi].sai.sin_family=AF_INET;
memmove(&wire[wi].sai.sin_addr,&rconf.txaddr,sizeof(struct in_addr));
wire[wi].sai.sin_port=htons(rconf.txport);
// Initiate connection
int cres = connect(wire[wi].skt,(struct sockaddr *)&wire[wi].sai,sizeof(struct sockaddr_in));
// Comprehend results
if (cres==0)
{ // Connect already established
wire[wi].state=CONNECTED;
return 0;
};
if (cres==-1) // Socket error meaning depends on errno
switch (errno)
{
// This is OK for non-blocking sockets
case EINPROGRESS : { // Connect initiated
wire[wi].state=CONNECTING;
return 0;
};
default : return TEC_SOCK_ERR;
};
return 0; // Should not normally happen

显然,EINPROGRESS 是这里最常见的情况,因此循环中还有另一个代码

            case CONNECTING :   {
struct timeval tv;
fd_set rfds;
FD_ZERO(&rfds);
FD_SET(wire[wc].skt,&rfds);
memset(&tv,0,sizeof(struct timeval));
int retval = select(1,NULL,&rfds,NULL,&tv);
printf("%d:%d ",retval,errno);
fflush(stdout);
// Done
break;
};

我不明白的是为什么这里的 select 总是返回 0 和 EAGAIN,无论连接是否成功。

我正在寻找最可靠的方法来判断非阻塞套接字上的连接是否成功。提前致谢。

更新。似乎没有明显的错误,所以我只会做一个 MWE,最终在其他地方发现它是愚蠢的:)

最佳答案

      int retval = select(1,NULL,&rfds,NULL,&tv);

select的第一个参数不是fd的数量,而是(来自linux man page):

nfds is the highest-numbered file descriptor in any of the three sets, plus 1.

除非您的套接字有 fd 0(在这种情况下您必须关闭 stdin),否则您使用的值 1 是错误的。这意味着集合中小于 nfds 的所有 fds(即没有)都已准备好,因此 select 返回 0。代码应改为:

       int retval = select(  1+wire[wc].skt,  NULL,&rfds,NULL,&tv);

关于c - 零超时的非阻塞 connect() 和 select() 返回 0,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/29602744/

26 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com