WIN网络编程-WSAEventSelect

本文介绍如何使用WSAEventSelect模型实现网络事件处理。通过创建事件对象与套接字关联,利用WSAWaitForMultipleEvents等待多个事件,进而处理如连接请求、读写等网络事件。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

//它与WSAAsyncSelect模型类似是因为它也接收FD_XXX类型的网络事件,是
//经由事件对象句柄通知。
//为一组网络事件创建一个事件对象,再调用WSAEventSelect函数将网络事件
//和事件对象关联起来,当网络事件发生是,WINSOCK使相应的事件对象受信,
//在事件对象上等待函数WSAWaitForMultipleEvents就会返回。然后,调
//用WSAEnumNetworkEvents获取网络事件。
//WSAWaitForMultipleEvents最多支持WSA_MAXIMUM_WAIT_EVENTS=64个对象。
//WSANETWORKEVENTS结构
//lNetworkEvents指定已经发生的网络事件
//iErrorCode[FD_MAX_EVENTS]指定与lNetworkEvents相关的出错代码,未出错为0
//具体使用WSAAsyncSelect模型步骤如下:
//1)创建一个事件句柄表和一个对应的套接字句柄表
//2)每创建一个套接字,就创建一个事件对象,把它们的句柄分别放入上
//面的两个表中,调用WSAAsyncSelect增加它们的关联
//3)调用WSAWaitForMultipleEvents在所有事件对象上等待,返回后,对事
//件句柄表中每个事件调用WSAWaitForMultipleEvents确认哪些套接字有网络
//事件发生。
//4)处理发生的网络事件,继续在事件对象上等待

// initsock.h

#include<winsock2.h>
#pragma comment(lib, "WS2_32")

class CInitSock
{
public:
CInitSock();
~CInitSock();
};

inline CInitSock::CInitSock()
{
// 初始化WS2_32.dll
WSADATA wsaData;
WORD sockVersion = MAKEWORD(2, 2);
if(::WSAStartup(sockVersion,&wsaData) != 0)
{
exit(0);
}
}

inline CInitSock::~CInitSock()
{
::WSACleanup();
}

//////////////////////////////////////////////////
// WSAEventSelect文件

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

// 初始化Winsock库
CInitSock theSock;

int main()
{
// 事件句柄和套节字句柄表
WSAEVENTeventArray[WSA_MAXIMUM_WAIT_EVENTS];
SOCKETsockArray[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函数会自动重置受信事件
WSANETWORKEVENTSevent;
::WSAEnumNetworkEvents(sockArray[i],eventArray[i], &event);
if(event.lNetworkEvents&FD_ACCEPT)//处理FD_ACCEPT通知消息
{
if(event.iErrorCode[FD_ACCEPT_BIT]== 0)
{
if(nEventTotal> WSA_MAXIMUM_WAIT_EVENTS)
{
printf("Too many connections! \n");
continue;
}
SOCKETsNew = ::accept(sockArray[i], NULL, NULL);
WSAEVENTevent = ::WSACreateEvent();
::WSAEventSelect(sNew,event, FD_READ|FD_CLOSE|FD_WRITE);
//添加到表中
eventArray[nEventTotal]= event;
sockArray[nEventTotal]= sNew;
nEventTotal++;
}
}
elseif(event.lNetworkEvents &FD_READ)//处理FD_READ通知消息
{
if(event.iErrorCode[FD_READ_BIT]== 0)
{
charszText[256];
intnRecv = ::recv(sockArray[i], szText, strlen(szText), 0);
if(nRecv>0)
{
szText[nRecv]= '\0';
printf("接收到数据:%s\n", szText);
}
}
}
elseif(event.lNetworkEvents &FD_CLOSE)//处理FD_CLOSE通知消息
{
if(event.iErrorCode[FD_CLOSE_BIT]== 0)
{
::closesocket(sockArray[i]);
for(intj=i; j<nEventTotal-1; j++)
{
sockArray[j]= sockArray[j+1];
sockArray[j]= sockArray[j+1];
}
nEventTotal--;
}
}
elseif(event.lNetworkEvents &FD_WRITE)//处理FD_WRITE通知消息
{
}
}
}
}
return 0;
}

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值