windows 网络编程经典入门的问题
2010-04-03 18:54
639 查看
我现在的情况是OnAccept()中的异步事件注册不成功,用WSAGetLastError()获得的错误代码为0,用switch无法获得错误消息,下面是原代码,我的代码是在其基础上完善的,框架大致是一样的,
其实我想知道的是accept参数中的Socket参数是用server(正在监听的参数)还是用消息映射函数从Windows中获得的SOCKET pSock = (SOCKET)wParam 这个参数,我使用server时无法重新注册异步事件,但使用pSock却可以,但我根本不知道用这个pSock取得的SOCKET对象是否是合法的,虽然注册异步事件成功,但客户端断开连接时,服务端仍无任何响应,IP_EVENT 事件我已重新注册
到消息映射中去,用另一个消息映射函数 IPEvent() 与其关联,消息映射是正常合法的
#define NETWORK_EVENT USER_MESSAGE+100 //定义网络事件
sockaddr_in clientaddr; file: //暂时存放客户端IP地址
//自己定义消息映射函数,将上面定义的网络事件映射到处理函数
//OnNetEvent为网络事件处理函数,它在下面定义
ON_MESSAGE(NETWORK_EVENT, OnNetEvent);
在你对话框中的初始化函数中调用下面的初始化网络的子函数
BOOL InitNetwork() //初始化网络
{
//初始化TCP协议
BOOL ret = WSAStartup(MAKEWORD(2, 2), &wsaData);
if (ret != 0)
{
MessageBox("初始化套接字失败!");
return FALSE;
}
//创建服务器端套接字
SOCKET serverSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
if (serverSocket == INVALID_SOCKET)
{
MessageBox("创建套接字失败!");
closesocket(m_Socket);
WSACleanup();
return FALSE;
}
//绑定到本地一个端口上
sockaddr_in localaddr;
localaddr.sin_family = AF_INET;
localaddr.sin_port = htons(1688);
localaddr.sin_addr.s_addr = 0;
if (bind(serverSocket , (const struct sockaddr*)&localaddr,
sizeof(sockaddr)) == SOCKET_ERROR)
{
MessageBox("绑定地址失败!");
closesocket(m_Socket);
WSACleanup();
return FALSE;
}
//注册网络异步事件,m_hWnd为应用程序的主对话框或主窗口的句柄
WSAAsyncSelect(serverSocket, m_hWnd, NETWORK_EVENT,
FD_ACCEPT | FD_CLOSE | FD_READ | FD_WRITE);
listen(serverSocket, 5); //设置侦听模式
return TRUE;
}
//定义网络事件的响应函数
void OnNetEvent(WPARAM wParam, LPARAM lParam)
{
//调用API函数,得到网络事件类型
int iEvent = WSAGETSELECTEVENT(lParam);
//得到发出此事件的客户端套接字
SOCKET pSock = (SOCKET)wParam;
switch (iEvent)
{
case FD_ACCEPT: //客户端连接请求
{
OnAccept();
break;
}
case FD_CLOSE: //客户端断开事件:
{
OnClose(pSock);
break;
}
case FD_READ: //网络数据包到达事件
{
OnReceive(pSock);
break;
}
case FD_WRITE: //发送网络数据事件
{
OnSend(pSock);
break;
}
default: break;
}
}
void OnAccept(SOCET pSock) //响应客户端连接请求函数
{
int len = sizeof(sockaddr);
//调用API函数,接受连接,并返回一个新套接字
//还可以获得客户端的IP地址
SOCKET clientSocket = accept(serverSocket,
(struct sockaddr*) & clientaddr, &len);
//为新的socket注册异步事件,注意没有Accept事件
if (WSAAsyncSelect(clientSocket , m_hWnd, IP_EVENT,
FD_CLOSE | FD_READ | FD_WRITE) == SOCKET_ERROR)
{
MessageBox("注册异步事件失败!");
return ;
}
//自编函数,将此客户端的相关信息保存下来:套接字、
// IP地址、登陆时间
saveClientSocket(clientSocket, clientAddr,currentTimer);
}
void OnClose(SOCET pSock)
{
//自编函数,结束与相应的客户端的通信,释放相应资源并做相应处理
endClientSocket(pSock);
}
void OnSend(SOCET pSock)
{
//自编函数,在给客户端发数据时做一些预处理
handleOnSend(pSock);
}
void OnReceive(SOCET pSock)
{
recv(); //调用API函数,读出网络缓冲区中的数据包
//自编函数,将此数据包和发出此数据的客户端
//clientSocket封装成一条网络消息
buildNetMsg();
//自编函数,将此网络消息放入一个消息队列中,由工作线程去处理
saveNetMsg();
SetEvent(); //用事件对象触发工作线程
}
其实我想知道的是accept参数中的Socket参数是用server(正在监听的参数)还是用消息映射函数从Windows中获得的SOCKET pSock = (SOCKET)wParam 这个参数,我使用server时无法重新注册异步事件,但使用pSock却可以,但我根本不知道用这个pSock取得的SOCKET对象是否是合法的,虽然注册异步事件成功,但客户端断开连接时,服务端仍无任何响应,IP_EVENT 事件我已重新注册
到消息映射中去,用另一个消息映射函数 IPEvent() 与其关联,消息映射是正常合法的
#define NETWORK_EVENT USER_MESSAGE+100 //定义网络事件
sockaddr_in clientaddr; file: //暂时存放客户端IP地址
//自己定义消息映射函数,将上面定义的网络事件映射到处理函数
//OnNetEvent为网络事件处理函数,它在下面定义
ON_MESSAGE(NETWORK_EVENT, OnNetEvent);
在你对话框中的初始化函数中调用下面的初始化网络的子函数
BOOL InitNetwork() //初始化网络
{
//初始化TCP协议
BOOL ret = WSAStartup(MAKEWORD(2, 2), &wsaData);
if (ret != 0)
{
MessageBox("初始化套接字失败!");
return FALSE;
}
//创建服务器端套接字
SOCKET serverSocket = socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
if (serverSocket == INVALID_SOCKET)
{
MessageBox("创建套接字失败!");
closesocket(m_Socket);
WSACleanup();
return FALSE;
}
//绑定到本地一个端口上
sockaddr_in localaddr;
localaddr.sin_family = AF_INET;
localaddr.sin_port = htons(1688);
localaddr.sin_addr.s_addr = 0;
if (bind(serverSocket , (const struct sockaddr*)&localaddr,
sizeof(sockaddr)) == SOCKET_ERROR)
{
MessageBox("绑定地址失败!");
closesocket(m_Socket);
WSACleanup();
return FALSE;
}
//注册网络异步事件,m_hWnd为应用程序的主对话框或主窗口的句柄
WSAAsyncSelect(serverSocket, m_hWnd, NETWORK_EVENT,
FD_ACCEPT | FD_CLOSE | FD_READ | FD_WRITE);
listen(serverSocket, 5); //设置侦听模式
return TRUE;
}
//定义网络事件的响应函数
void OnNetEvent(WPARAM wParam, LPARAM lParam)
{
//调用API函数,得到网络事件类型
int iEvent = WSAGETSELECTEVENT(lParam);
//得到发出此事件的客户端套接字
SOCKET pSock = (SOCKET)wParam;
switch (iEvent)
{
case FD_ACCEPT: //客户端连接请求
{
OnAccept();
break;
}
case FD_CLOSE: //客户端断开事件:
{
OnClose(pSock);
break;
}
case FD_READ: //网络数据包到达事件
{
OnReceive(pSock);
break;
}
case FD_WRITE: //发送网络数据事件
{
OnSend(pSock);
break;
}
default: break;
}
}
void OnAccept(SOCET pSock) //响应客户端连接请求函数
{
int len = sizeof(sockaddr);
//调用API函数,接受连接,并返回一个新套接字
//还可以获得客户端的IP地址
SOCKET clientSocket = accept(serverSocket,
(struct sockaddr*) & clientaddr, &len);
//为新的socket注册异步事件,注意没有Accept事件
if (WSAAsyncSelect(clientSocket , m_hWnd, IP_EVENT,
FD_CLOSE | FD_READ | FD_WRITE) == SOCKET_ERROR)
{
MessageBox("注册异步事件失败!");
return ;
}
//自编函数,将此客户端的相关信息保存下来:套接字、
// IP地址、登陆时间
saveClientSocket(clientSocket, clientAddr,currentTimer);
}
void OnClose(SOCET pSock)
{
//自编函数,结束与相应的客户端的通信,释放相应资源并做相应处理
endClientSocket(pSock);
}
void OnSend(SOCET pSock)
{
//自编函数,在给客户端发数据时做一些预处理
handleOnSend(pSock);
}
void OnReceive(SOCET pSock)
{
recv(); //调用API函数,读出网络缓冲区中的数据包
//自编函数,将此数据包和发出此数据的客户端
//clientSocket封装成一条网络消息
buildNetMsg();
//自编函数,将此网络消息放入一个消息队列中,由工作线程去处理
saveNetMsg();
SetEvent(); //用事件对象触发工作线程
}
相关文章推荐
- windows 网络编程经典入门
- Windows下C语言网络编程快速入门
- 经典编程书籍(C++, 网络, Windows, Linux)【转载】
- VC网络编程-Winsock开发网络通信程序的经典入门
- Windows网络编程入门:简单的客户端和服务器通信程序调试
- Windows下C语言网络编程快速入门
- 经典编程书籍(C++, 网络, Windows, Linux)
- 经典编程书籍(C++, 网络, Windows, Linux)
- Windows网络编程入门:简单的客户端和服务器通信程序调试
- Windows下C语言网络编程快速入门
- 经典编程书籍(C++, 网络, Windows, Linux)
- Windows网络编程细节问题
- Windows网络编程入门:简单的客户端和服务器通信程序调试
- 经典编程书籍(C++, 网络, Windows, Linux)
- 网络编程入门(windows&linux)
- WINDOWS编程入门一个初级问题的分析
- Windows网络编程入门:简单的客户端和服务器通信程序调试
- windows下C网络编程入门
- Windows下C语言网络编程快速入门
- 网络编程入门(windows&linux)