在Windows上为套接字设置recv的超时

时间:2016-09-07 16:52:34

标签: c++ sockets winsock

小时, 我认为我的代码是正确的,但它不起作用:(

要在Windows上设置recv功能的超时,我知道我必须使用此代码:

                    DWORD timeout = 2000;

                 if (setsockopt(listenSocket, SOL_SOCKET, SO_RCVTIMEO, (char*)&timeout, sizeof(DWORD)))
                 { perror("setsockopt");
                return -1;
                 }

但它没有用。

我服务器的代码是:

    SOCKET listenSocket;
SOCKET remoteSocket= INVALID_SOCKET;
SOCKADDR_IN Server_addr;
SOCKADDR_IN Client_addr;
int sin_size;
short port;

int wsastartup;
int ls_result;
WORD wVersionRequested = 0x0202;
WSADATA wsaData;

wsastartup = WSAStartup(wVersionRequested, &wsaData);
if (wsastartup != NO_ERROR) cout << "Errore WSAStartup()" << endl;

listenSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);

port = 4000;
Server_addr.sin_family = AF_INET;
Server_addr.sin_addr.s_addr = inet_addr("127.0.0.1");

Server_addr.sin_port = htons(port);

if (bind(listenSocket,(LPSOCKADDR) &Server_addr,sizeof(struct sockaddr)) < 0) {
    cout << "Server: error bind." << endl;
closesocket(listenSocket);
return -1;
}

ls_result = listen(listenSocket, SOMAXCONN);

sin_size = sizeof(struct sockaddr_in);
remoteSocket = accept(listenSocket, (struct sockaddr *) &Client_addr, &sin_size);

// SET THE TIME OUT
DWORD timeout = 300;
if (setsockopt(remoteSocket, SOL_SOCKET, SO_RCVTIMEO, (char*)&timeout, sizeof(DWORD)))
{ perror("setsockopt");
    return -1;
}

int i=0;
while (i<50){
    t_start = clock();

    // when client receives the send below it wait 3 seconds and then trasmits the answer
    send(remoteSocket, "code of start transmission", sizeof("code of start transmission"), 0);

    recv_size=recv(remoteSocket, messaggio, sizeof(messaggio), 0);

    printf("time for read=  %f second \n", ((double)(end - t_start)) / CLOCKS_PER_SEC);

    i=i+1;
}

客户端收到消息&#34;开始传输代码&#34;从服务器,它等待3秒,然后aswer到服务器。 我希望读取的时间是300毫秒,recv_size&lt; 0,而不是recv_size&lt; 0但读取的时间大约是1.5秒(服务器等待客户端的消息)。我不明白为什么。

我在Windows上使用eclipse和mingw-w64。

请有人可以帮助我吗?

2 个答案:

答案 0 :(得分:1)

  

要在Windows上设置recv功能的超时,我知道我必须使用此代码:

             DWORD timeout = 2000;

             if (setsockopt(listenSocket, SOL_SOCKET, SO_RCVTIMEO, (char*)&timeout, sizeof(DWORD)))
             { perror("setsockopt");
            return -1;
             }

没有。它应该是int,而不是DWORD,但主要问题是你在这里设置accept()超时,因为这是监听套接字。您需要在接受的套接字上设置它。

答案 1 :(得分:1)

您的代码尝试在超时后使用套接字。这不是一个好主意,因为套接字仍处于失败阻塞操作的中间位置,并且无法启动新操作。没有办法解开之前完成的操作部分,并将套接字放回操作开始前的位置。

一旦阻塞套接字操作超时,您可以放心地关闭套接字。不支持撤消部分完成的操作并使套接字处于任何类型的理智状态。

  

如果发送或接收操作在套接字上超时,套接字状态是不确定的,不应使用[。] - MSDN

SO_RCVTIMEO套接字选项绝不应该用于设计用于套接字的代码中。它可以防止代码中的无限等待,这些代码本身并不适用于套接字。这些不是您正在寻找的机器人。