小时, 我认为我的代码是正确的,但它不起作用:(
要在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。
请有人可以帮助我吗?
答案 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套接字选项绝不应该用于设计用于套接字的代码中。它可以防止代码中的无限等待,这些代码本身并不适用于套接字。这些不是您正在寻找的机器人。