对于CAsyncSocket::Connect()太痛苦了!

时间:2022-12-13 17:47:58
我在使用CAsyncSocket进行TCP/IP通信时遇到一个很棘手的问题
我发现  我在连接时如果
连接的IP地址正确但是端口错误时  我立刻接收到OnConnect返回10061
连接的IP地址错误或者网络断开的话   我在45s之后才能收到OnConnect返回10060
但是对于我来说,在连接等待45s时可能需要立刻关闭现在的连接,
实际上我跟踪到VC的MFC代码发现,此时关闭Socket是可以关闭Socket的,但实际上
与CAsyncSocket相关联的隐藏窗口并没有Destroy,即
在CAsyncSocket::DetachHandle()中pState->m_pmapSocketHandle->IsEmpty()
返回了false
不知道该如何处理?
并且SetSockOpt不允许设置连接Timeout时间的,
请大家提一些建议!!

5 个解决方案

#1


#2


把CSDN与中文yahoo翻了底朝天,也没找到如何设置socket的连接超时的满意方法,问此问题的兄弟已有一大堆,这里偶就讲一下win下如何设置socket的connect超时。
设置connect的超时很简单,CSDN上也有人提到过使用select,但却没有一个令人满意与完整的答案。偶所讲的也正是select函数,此函数集成在winsock1.1中,简单点讲,"作用使那些想避免在套接字调用过程中被锁定的应用程序,采取一种有序的方式,同时对多个套接字进行管理"(《Windows网络编程技术》原话)。使用方法与解释请见《Windows网络编程技术》。
在使用此函数前,需先将socket设置为非锁定模式,这样,在connect时,才会立马跳过,同时,通常也会产生一个WSAEWOULDBLOCK错误,这个错误没关系。再执行select则是真正的超时。

WSADATA wsd;
SOCKET cClient;
int ret;
struct sockaddr_in server;
hostent *host=NULL;

if(WSAStartup(MAKEWORD(2,0),&wsd)){return 0;}
cClient=socket(AF_INET,SOCK_STREAM,IPPROTO_TCP);
if(cClient==INVALID_SOCKET){return 0;}
//set Recv and Send time out
int TimeOut=6000; //设置发送超时6秒
if(::setsockopt(cClient,SOL_SOCKET,SO_SNDTIMEO,(char *)&TimeOut,sizeof(TimeOut))==SOCKET_ERROR){
return 0;
}
TimeOut=6000;//设置接收超时6秒
if(::setsockopt(cClient,SOL_SOCKET,SO_RCVTIMEO,(char *)&TimeOut,sizeof(TimeOut))==SOCKET_ERROR){
return 0;
}
//设置非阻塞方式连接
unsigned long ul = 1;
ret = ioctlsocket(cClient, FIONBIO, (unsigned long*)&ul);
if(ret==SOCKET_ERROR)return 0;

//连接
server.sin_family = AF_INET;
server.sin_port = htons(25);
server.sin_addr .s_addr = inet_addr((LPCSTR)pSmtp);
if(server.sin_addr.s_addr == INADDR_NONE){return 0;}

connect(cClient,(const struct sockaddr *)&server,sizeof(server));

//select 模型,即设置超时
struct timeval timeout ;
fd_set r;

FD_ZERO(&r);
FD_SET(cClient, &r);
timeout.tv_sec = 15; //连接超时15秒
timeout.tv_usec =0;
ret = select(0, 0, &r, 0, &timeout);
if ( ret <= 0 )
{
::closesocket(cClient);
return 0;
}
//一般非锁定模式套接比较难控制,可以根据实际情况考虑 再设回阻塞模式
unsigned long ul1= 0 ;
ret = ioctlsocket(cClient, FIONBIO, (unsigned long*)&ul1);
if(ret==SOCKET_ERROR){
::closesocket (cClient);
return 0;
}

#3


现在其实是一个这样的问题:如果连接不上未知IP地址,我的程序一旦用户挂机,就会出错,
堆栈信息如下所示
系统是收到了一个消息  WM_SOCKET_NOTIFY  68  657850384
不知道问题的根源在哪儿?

CAsyncSocket::DoCallBack(unsigned int 68, long 657850384) line 512 + 25 bytes
CSocket::ProcessAuxQueue() line 822
CSocketWnd::OnSocketNotify(unsigned int 68, long 657850384) line 1126
CWnd::OnWndMsg(unsigned int 883, unsigned int 68, long 657850384, long * 0x0012fd90) line 1815 + 17 bytes
CWnd::WindowProc(unsigned int 883, unsigned int 68, long 657850384) line 1585 + 30 bytes
AfxCallWndProc(CWnd * 0x004810b0 {CSocketWnd hWnd=0x29760694}, HWND__ * 0x29760694, unsigned int 883, unsigned int 68, long 657850384) line 215 + 26 bytes
AfxWndProc(HWND__ * 0x29760694, unsigned int 883, unsigned int 68, long 657850384) line 368
AfxWndProcBase(HWND__ * 0x29760694, unsigned int 883, unsigned int 68, long 657850384) line 220 + 21 bytes
USER32! 77e41777()

#4


??

#5


CSocket是否把设置非阻塞的属性屏蔽了?我在CSocket::IOCtl()根本没有用,还是要等待大约45s才能connect()返回

#1


#2


把CSDN与中文yahoo翻了底朝天,也没找到如何设置socket的连接超时的满意方法,问此问题的兄弟已有一大堆,这里偶就讲一下win下如何设置socket的connect超时。
设置connect的超时很简单,CSDN上也有人提到过使用select,但却没有一个令人满意与完整的答案。偶所讲的也正是select函数,此函数集成在winsock1.1中,简单点讲,"作用使那些想避免在套接字调用过程中被锁定的应用程序,采取一种有序的方式,同时对多个套接字进行管理"(《Windows网络编程技术》原话)。使用方法与解释请见《Windows网络编程技术》。
在使用此函数前,需先将socket设置为非锁定模式,这样,在connect时,才会立马跳过,同时,通常也会产生一个WSAEWOULDBLOCK错误,这个错误没关系。再执行select则是真正的超时。

WSADATA wsd;
SOCKET cClient;
int ret;
struct sockaddr_in server;
hostent *host=NULL;

if(WSAStartup(MAKEWORD(2,0),&wsd)){return 0;}
cClient=socket(AF_INET,SOCK_STREAM,IPPROTO_TCP);
if(cClient==INVALID_SOCKET){return 0;}
//set Recv and Send time out
int TimeOut=6000; //设置发送超时6秒
if(::setsockopt(cClient,SOL_SOCKET,SO_SNDTIMEO,(char *)&TimeOut,sizeof(TimeOut))==SOCKET_ERROR){
return 0;
}
TimeOut=6000;//设置接收超时6秒
if(::setsockopt(cClient,SOL_SOCKET,SO_RCVTIMEO,(char *)&TimeOut,sizeof(TimeOut))==SOCKET_ERROR){
return 0;
}
//设置非阻塞方式连接
unsigned long ul = 1;
ret = ioctlsocket(cClient, FIONBIO, (unsigned long*)&ul);
if(ret==SOCKET_ERROR)return 0;

//连接
server.sin_family = AF_INET;
server.sin_port = htons(25);
server.sin_addr .s_addr = inet_addr((LPCSTR)pSmtp);
if(server.sin_addr.s_addr == INADDR_NONE){return 0;}

connect(cClient,(const struct sockaddr *)&server,sizeof(server));

//select 模型,即设置超时
struct timeval timeout ;
fd_set r;

FD_ZERO(&r);
FD_SET(cClient, &r);
timeout.tv_sec = 15; //连接超时15秒
timeout.tv_usec =0;
ret = select(0, 0, &r, 0, &timeout);
if ( ret <= 0 )
{
::closesocket(cClient);
return 0;
}
//一般非锁定模式套接比较难控制,可以根据实际情况考虑 再设回阻塞模式
unsigned long ul1= 0 ;
ret = ioctlsocket(cClient, FIONBIO, (unsigned long*)&ul1);
if(ret==SOCKET_ERROR){
::closesocket (cClient);
return 0;
}

#3


现在其实是一个这样的问题:如果连接不上未知IP地址,我的程序一旦用户挂机,就会出错,
堆栈信息如下所示
系统是收到了一个消息  WM_SOCKET_NOTIFY  68  657850384
不知道问题的根源在哪儿?

CAsyncSocket::DoCallBack(unsigned int 68, long 657850384) line 512 + 25 bytes
CSocket::ProcessAuxQueue() line 822
CSocketWnd::OnSocketNotify(unsigned int 68, long 657850384) line 1126
CWnd::OnWndMsg(unsigned int 883, unsigned int 68, long 657850384, long * 0x0012fd90) line 1815 + 17 bytes
CWnd::WindowProc(unsigned int 883, unsigned int 68, long 657850384) line 1585 + 30 bytes
AfxCallWndProc(CWnd * 0x004810b0 {CSocketWnd hWnd=0x29760694}, HWND__ * 0x29760694, unsigned int 883, unsigned int 68, long 657850384) line 215 + 26 bytes
AfxWndProc(HWND__ * 0x29760694, unsigned int 883, unsigned int 68, long 657850384) line 368
AfxWndProcBase(HWND__ * 0x29760694, unsigned int 883, unsigned int 68, long 657850384) line 220 + 21 bytes
USER32! 77e41777()

#4


??

#5


CSocket是否把设置非阻塞的属性屏蔽了?我在CSocket::IOCtl()根本没有用,还是要等待大约45s才能connect()返回