选择TCP套接字超时

时间:2013-06-12 17:09:56

标签: sockets select timeout connect

我对此代码有疑问:

FD_ZERO(&cset);
FD_SET(s, &cset);

tval.tv_sec = TIMEOUT; 
tval.tv_usec = 0; 

n = select(FD_SETSIZE, &cset, NULL, NULL, &tval);

if (n==-1) {
    printf(" select() failed \n");
    exit(-1);
}
if (n>0) {
    check_control = connect(s,(struct sockaddr*)
    &indirizzo_remoto,sizeof(indirizzo_remoto));

    if (check_control == -1) {
        printf("Errore connect()\n");
    }

}else{
    printf("Timeout. I'll shutdown the client");
    exit(-1);
}

我想为连接插入超时但它不起作用:

我使用了正确的服务器IP地址和端口号,但连接超时。

非常感谢您的帮助。

1 个答案:

答案 0 :(得分:3)

您正在使用select()检查给定套接字是否处于可读状态,然后再在同一套接字上调用connect()。那永远不会奏效。未连接的TCP套接字永远不会处于可读状态,并且在select()首先被调用之前不能与connect()一起使用。

实现connect()调用超时的唯一方法是首先将套接字置于非阻塞模式(默认情况下套接字阻塞),然后调用connect()(返回EINPROGRESS 1}}如果套接字尝试连接时出错,然后使用select()等待套接字进入可写状态,表明连接成功,或者错误状态,表示连接失败。

试试这个:

fcntl(s, F_SETFL, O_NONBLOCK);

或者:

flags = 1;
ioctl(s, FIOBIO, &flags);

取决于您的平台。

然后:

check_control = connect(s, (struct sockaddr*) &indirizzo_remoto, sizeof(indirizzo_remoto));
if (check_control == -1)
{
    if (errno != EINPROGRESS)
    {
        printf("Errore connect()\n");
        exit(-1);
    }

    FD_ZERO(&wset);
    FD_SET(s, &wset);

    FD_ZERO(&eset);
    FD_SET(s, &eset);

    tval.tv_sec = TIMEOUT; 
    tval.tv_usec = 0; 

    n = select(s+1, NULL, &wset, &eset, &tval);
    if (n == -1)
    {
        printf(" select() failed \n");
        exit(-1);
    }

    if (n == 0)
    {
        printf("Timeout. I'll shutdown the client");
        exit(-1);
    }

    if (FD_ISSET(s, &eset))
    {
        printf("Cannot connect. I'll shutdown the client");
        exit(-1);
    }

    int err = -1;
    getsockopt(s, SOL_SOCKET, SO_ERROR, &err, sizeof(err));
    if (err != 0)
    {
        printf("Cannot connect. I'll shutdown the client");
        exit(-1);
    }
}

// connected...