首先说明什么是Windows套接字模式.
其分为两类:阻塞模式/非阻塞模式.
阻塞模式:I/O操作完成前执行操作的WinSock调用会一直等候下去,不会立即返回到程序中.
非阻塞模式:WinSock函数无论如何都会立即返回.
对阻塞套接字
他的一个缺点在于,应用程序很难同时通过多个建好连接的套接字通信,使用前述的方案,可对应用程序进行修改,令其为连接好的每个套接字都分配一个读线程,以及一个数据处理线程.尽管这仍然会增大一些开销,但的确是一种可行的方案.唯一的缺点便是伸缩性不好,以后想同时处理大量套接字时就不太方便.
对非阻塞模式
可以将前面文章讲到的套接字(阻塞)设置为非阻塞套接字模式
nret = ioctlsocket(s,FIONRIO,(unsigned Long*)&ul);
if(nret == SOCKET_ERROR)
{
//未能将套接字置入非阻塞模式
}
阻塞和非阻塞套接字模式都存在着优点和缺点.
所以考虑使用套接字I/O模型,他将帮助应用程序通过一种异步方式,同时对一个或多个套接字上进行的通信加以管理.
套接字I/O模型
总共包括6中模型
1.阻塞模型
2.select模型
3.WSAAsyncSelect(异步选择)
要想使用WSAAsyncSelect模型,在应用程序中,首先必须用CreateWindow函数创建一个窗口,再为该窗口提供一个窗口过程支持函数(Winproc).也可使用一个对话框,为其提供一个对话过程来代替窗口过程,这是因为对话框本质也是窗口.
int WSAAsyncSelect(
SOCKET s,
HWND hwnd,
unsighed int wMsg,
long lEvent
)
//s参数在服务器端就是监听套接字
//hWnd参数指定的是一个窗口句柄,即上面建立的新的窗口,它标识的是网络事件发生之后,想要收到通知消息的那个窗口或对话框
//wMsg参数指定在发生网络事件时,打算接收的消息.该消息将被投递到由hWnd窗口句柄所标识的那个窗口.通常,应用程序需要将这个消息设为比Windows的WM_USER大的一个值,以免网络窗口消息与预定义的标准窗口消息发生混淆.
//`# define WM_SOCKET WM_USER+1`
//IEvent,是一个位掩码,指定一系列网络事件的组合,到底是用什么事件组合要取决于应用程序的身份是服务器还是客户机,多个事件类型用OR(或)运算
例子:
WSAAsyncSelect(s, hwnd,WM_SOCKET,FD_CONNECT|FD_READ|FD_WRITE|FD_CLOSE);
注意:
1.多个事件务必在套接字上一次完成注册;
2. 一旦在某个套接字上启用了事件通知,那么以后除非明确调用closesocket命令\或者由应用程序针对这个套接字调用了WSAAsyncSelect,从而更改注册的网络事件类型,否则,事件通知总是有效的;
3.若参数IEvent设为0,则相当于停止在套接字上进行的所有网络事件通知
应用程序在一个套接字上成功调用了WSAAsyncSelect之后,它会与hWnd窗口句柄参数相关联的窗口过程中,以Windows消息的形式,接受网络事件通知
窗口过程通常定义
LRESULT CALLBACK WindowProc(
HWND hWnd,//那个对话窗口
UINT wMsg,//WSAAsyncSelect调用中定义的消息(哪个窗口的?我理解是),是哪个窗口的消息
WPARAM wParam,//一个套接字,该套接字上发生了一个网络事件
LPARAM lParam//包含两方面重要信息,低字节指定了已经发生的网络事件,高字节包含了可能出现的任何错误代码
)
//当网络事件消息抵达窗口过程后,应用程序首先应检查lParam的高字节,以判断套接字上是否发生了网络错误,这里用一个宏:***WSAGETSELECTERROR(lParam)***可用它返回高字节包含的错误信息.如果没错误的话接着便应检查低字节内容用另一个宏***WSAGETSELECTEVENT(lParam)***
下面是一个示例
# define WM_SOCKET WM_USER+1 // hwnd句柄标识的那个窗口号
# include<winsock2.h>
# include<windows.h>
int main()
{
WSADATA wsadata;
SOCKET listens;
SOCKADDR_IN addrSer;
HWND window;
WSAStartup(MAKEWORD(2,2),&wsadata);
listens = socket(AF_INET, SOCK_STREAM,IPPROTO_TCP);
addrSer.sin_family = AF_INET;
addrSer.sin_addr.s_addr = htonl(INADDR_ANY)