FD_ACCEPT检测到,但然后调用'接受'失败,10038 - WSAENOTSOCK
基本上,就是这样。同时,客户的连接呼叫是成功的。这怎么可能?我没有添加任何代码,因为我不知道错误在哪里。FD_ACCEPT检测到,但然后调用'接受'失败,10038 - WSAENOTSOCK
服务器:检测FD_ACCEPT。呼叫接受()失败。
客户端:呼叫连接()成功。然后它检测到FD_CONNECT。以下send()成功。之后的send()失败(10053 - WSAECONNABORTED)。
void Server::get_addressinfo() {
// Resolve the local address and port to be used by the server
const char * p = port.c_str();
int iResult = getaddrinfo(NULL, p, &hints, &result);
if (iResult != 0) {
throw MyException("getaddrinfo failed.");
}
}
void Server::create_socket() {
ListenSocket = socket(result->ai_family, result->ai_socktype, result->ai_protocol);
if (ListenSocket == INVALID_SOCKET) {
throw MyException("socket creation failed.");
}
}
void Server::bind_socket() {
int iResult = bind(ListenSocket, result->ai_addr, (int) result->ai_addrlen);
if (iResult == SOCKET_ERROR) {
closesocket(ListenSocket);
throw MyException("bind failed.");
}
}
void Server::listen_for_connection() {
int iResult = listen(ListenSocket, SOMAXCONN);
if (iResult == SOCKET_ERROR) {
closesocket(ListenSocket);
throw MyException("listen failed.");
}
}
void Server::goLive() {
get_addressinfo();
create_socket();
bind_socket();
listen_for_connection();
wait_for(FD_ACCEPT);
accept_connection();
}
void Server::wait_for(u_int event_type) {
WSAEVENT event = WSACreateEvent();
WSAEventSelect(ListenSocket, event, event_type);
WSAEVENT lphEvents[1] = {event};
WSANETWORKEVENTS NetworkEvents = {0};
int nReturnCode = WSAWaitForMultipleEvents(1, &lphEvents[0], false, WSA_INFINITE, false);
if (nReturnCode==WSA_WAIT_FAILED)
throw MyException("WSA__WAIT_FAILED.\n");
if (WSAEnumNetworkEvents(ListenSocket, lphEvents[0], &NetworkEvents) == SOCKET_ERROR)
throw MyException("WSAEnumNetworkEvents => SOCKET_ERROR.\n");
WSACloseEvent(event); ***// THIS WAS THE BUG !!!***
}
void Server::accept_connection() {
ClientSocket = accept(ListenSocket, NULL, NULL);
if (ClientSocket == INVALID_SOCKET) {
closesocket(ListenSocket);
throw MyException("accept failed.");
}
}
我在代码上做了一些改进,但是bug仍然存在。现在我发现了问题行:
WSACloseEvent(event);
注释掉后,服务器接受连接。
从MSDN WSACloseEvent函数文档:
的WSACloseEvent功能关闭句柄到一个事件对象并释放与该事件对象相关联的资源。
我想这意味着它释放了socket =>因此10038 - WSAENOTSOCK。我落入了与INVALID_SOCKET不相等的套接字值。所以我认为套接字是有效的。它不是......
你应该删除整个事件的事情。你不需要它。 accept()方法已经被阻塞。如果您需要计时,请使用SO_RCVTIMEO。 – EJP
我可以重现错误。这很奇怪,因为'WSACreateEvent()'简单地调用'CreateEvent()','WSAWaitForMultipleEvents()'简单地调用'WaitForMultipleObjectsEx()',而'WSACloseEvent()'简单地调用'CloseHandle()'。所以你实际上关闭了一个标准的Win32事件,这与套接字无关。另一方面,我猜'WSAEventSelect()'是以某种方式将它们连接在一起的,所以关闭一个关闭另一个。 –
@EJP:或者,你可以在调用accept()之前使用'select()'。 –
错误信息几乎不可能更清晰,'错误在哪里'显然是在调用'accept()'。但如果你不想要答案,不要发布代码。随你便。 – EJP
添加了代码。 – Earnie
如果接受失败,为什么要关闭侦听套接字?这是否应该只接受一个连接或者是处理多个连接的方案的一部分? –