WSAEventSelect 模型


Winsock提供了另一种有用的异步事件通知 I/O模型——WSAEventSelect模型。这个模型与 WSAAsyncSelect模型类似,允许应用程序在一个或者多个套接字上接收基于事件的网络通知。它与 WSAAsyncSelect 模型类似是因为它也接收FD_XXX 类型的网络事件,不过并不是依靠Windows 的消息驱动机制,而是经由事件对象句柄通知。 



//
// WSAEventSelect

#include "../common/initsock.h"
#include <stdio.h>
#include <iostream.h>
#include <windows.h>

CInitSock initSock;		// 初始化Winsock库

int main()
{
	// 事件句柄和套接字句柄表
	WSAEVENT	eventArray[WSA_MAXIMUM_WAIT_EVENTS];
	SOCKET		sockArray[WSA_MAXIMUM_WAIT_EVENTS];
	int nEventTotal = 0;

	USHORT nPort = 4567;	// 服务器监听端口

	// 创建监听套接字
	SOCKET sListen = ::socket(AF_INET, SOCK_STREAM, IPPROTO_TCP);
	sockaddr_in sin;
	sin.sin_family = AF_INET;
	sin.sin_port = htons(nPort);
	sin.sin_addr.S_un.S_addr = INADDR_ANY;
	
	// 绑定端口
	if (::bind(sListen, (sockaddr *)&sin, sizeof(sin)) == SOCKET_ERROR) 
	{
		printf("Failed bind() \n");
		return -1;
	}
	
	// 进入监听
	::listen(sListen, 5);
	
	// 创建事件对象,并关联到新的套接字
	WSAEVENT event = ::WSACreateEvent();
	// 将网络事件和事件对象关联起来;
	::WSAEventSelect(sListen, event, FD_ACCEPT|FD_CLOSE);	

	// 添加到表中
	eventArray[nEventTotal] = event;
	sockArray[nEventTotal] = sListen;
	nEventTotal++;

	// 处理网络事件
	while (TRUE) 
	{
		// 在所有事件对象上等待,下面函数会等待网络事件发生
		int nIndex = ::WSAWaitForMultipleEvents(nEventTotal, eventArray, FALSE, WSA_INFINITE, FALSE);
		
		// 对每个事件再次调用WSAWaitForMultipleEvents, 以便确定它们的状态;
		nIndex = nIndex - WSA_WAIT_EVENT_0;
		for(int i = nIndex; i <nEventTotal; i++)
		{
			nIndex = ::WSAWaitForMultipleEvents(1, &eventArray[i], TRUE, 1000, FALSE);
			if (nIndex == WSA_WAIT_FAILED || nIndex == WSA_WAIT_TIMEOUT) 
			{
				continue;
			}
			else
			{
				// 获取到来的通知消息,WSAEnumNetworkEvents函数会自动重置受信事件;
				// 找到与之对应的套接字,WSAEnumNetworkEvents查看发生了什么网络事件;
				WSANETWORKEVENTS event;
				::WSAEnumNetworkEvents(sockArray[i], eventArray[i], &event);

				if (event.lNetworkEvents & FD_ACCEPT)	// 处理FD_ACCEPT通知消息
				{
					if (event.iErrorCode[FD_ACCEPT_BIT] == 0)	// FD_ACCEPT未出错
					{
						if (nEventTotal > WSA_MAXIMUM_WAIT_EVENTS) 
						{
							printf(" Too many connections");
							continue;
						}

						SOCKET sNew = ::accept(sockArray[i], NULL, NULL);

						WSAEVENT event = ::WSACreateEvent();
						::WSAEventSelect(sNew, event, FD_READ|FD_CLOSE|FD_WRITE);
						eventArray[nEventTotal] = event;
						sockArray[nEventTotal] = sNew;
						nEventTotal++;
					}
				}
				else if (event.lNetworkEvents & FD_READ)	// 处理FD_READ通知消息
				{
					if (event.iErrorCode[FD_READ_BIT] == 0) 
					{
						char szText[256];
						int nRecv = ::recv(sockArray[i], szText, strlen(szText), 0);
						if (nRecv > 0) 
						{
							szText[nRecv] = '\0';
							printf("接受到数据:%s \n", szText);
						}
					}
				}
				else if (event.lNetworkEvents & FD_CLOSE)	// 处理FD_CLOSE通知消息  
				{
					if (event.iErrorCode[FD_CLOSE_BIT] == 0) 
					{
						::closesocket(sockArray[i]);

						for(int j = i; j < nEventTotal; j++)	// 删除数组元素;
						{
							sockArray[j] = sockArray[j+1];
						}
						nEventTotal--;
					}
				}
				else if (event.lNetworkEvents & FD_WRITE)	//  处理FD_WRITE通知消息  
				{
				}
			}
		}
	}

	return 0;
}




  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值