[C++ 网络协议] 异步通知I/O模型

1.什么是异步通知I/O模型

如图是同步I/O函数的调用时间流:

如图是异步I/O函数的调用时间流:

可以看出,同异步的差别主要是在时间流上的不一致。select属于同步I/O模型。epoll不确定是不是属于异步I/O模型,这个在概念上有些混乱,期望大佬的指点

这里说的异步通知I/O模型,实际上是select模型的改进方案。

2.实现异步通知I/O模型

2.1 实现异步通知I/O模型步骤

2.2 WSAEventSelect函数

#include<winsock2.h>

int WSAEventSelect(
SOCKET s,                //监视对象的套接字句柄
WSAEVENT hEventObject,   //传递事件对象句柄以验证事件发生与否
long lNetworkEvents      //监视的事件类型信息
);
成功返回0
失败返回SOCKET_ERROR

参数hEventObject:

#define WSAEVENT HANDLE

WSAEVENT就是HANDLE。

参数lNetworkEvents:

含义
FD_READ是否存在需要接收的数据
FD_WRITE能否以非阻塞的方式传输数据
FD_OOB是否收到带外数据
FD_ACCEPT是否有新的连接请求
FD_CLOSE是否有断开连接的请求

可以通过位或运算指定多个信息。

函数解释:

传入的套接字参数s,只要s发送lNetworkEvents事件,就会将hEventObject事件对象所指内核对象的状态,改为signaled状态。

与select函数的比较:

每个通过WSAEventSelect函数注册的套接字信息就已经注册到操作系统中了,这意味着,无需针对已注册的套接字重复调用WSAEventSelect。

还有一个实现方式是WSAAsyncSelect函数,使用这个函数时需要指定Windows句柄以获取发生的事件(跟UI有关)

2.3 创建WSAEVENT对象

创建manual-reset模式的事件对象。

方式一:

使用“windows中的线程同步”中所讲的CreateEvent函数。

方式二:

#include<winsock2.h>

WSAEVENT WSACreateEvent(void);
成功返回事件对象句柄
失败返回WSA_INVALID_EVENT

这种方式会直接创建manual-reset模式的事件对象。 其销毁函数:

#include<winsock2.h>

BOOL WSACloseEvent(WSAEVENT hEvent);
成功返回TRUE
失败返回FALSE

2.4 验证是否发生了事件

#include<winsock2.h>

DWORD WSAWaitForMultipleEvent(
DWORD cEvents,                //需要验证是否转为signaled状态的事件对象个数
const WSAEVENT* lphEvents,    //存有事件对象句柄的数组地址值
BOOL fWaitAll,                //TRUE,所有事件对象都在signaled状态时返回
                              //FALSE,只要其中1个变为signaled状态就返回
DWORD dwTimeout,              //以1/1000秒为单位指定超时,传递WSA_INFINITE时,直到signaled状态时才返回
                              //传递0时,表明不阻塞,是否是signaled状态都返回
BOOL fAlertable               //传递TRUE可进入alertable_wait(可警告等待)状态
);
成功:
返回值减去WSA_WAIT_EVENT_0时,可以得到第一个转变为signaled状态的事件对象句柄对应的索引,可在第二个参数中查找对应句柄。
超时则返回WSA_WAIT_TIMEOUT。
失败:
返回WSA_WAIT_FAILED(注意,原版书籍里这里打印错了)

最多可监视的事件对象数量为:WSA_MAXIMUM_WAIT_EVENTS常量。

要想监视更多,要么创建线程,要么扩展保存句柄的数组并多次调用这个函数。

注意:参数fwaitAll为FALSE时,是说只要其中1个变为signaled状态就返回,函数是返回了,但其有可能有多个事件对象变为了signaled状态

通过事件对象为manual-reset模式的特点,可以获取转为signaled状态的所有事件对象的句柄。

int start;
WSAEVENT events[num];
start=WSAWaitForMultipleEvents(num,events,FALSE,WSA_INFINITE,FALSE);
int first=start-WSA_WAIT_EVENT_0;
for(int i=first,i<num;++i)    //first是变为singaled状态的事件对象的索引的最小值
{
    //从第一个的signaled状态的事件对象开始,一个个判断是否siganled
    int sigEventIdx=WSAWaitForMultipleEvents(1,&events[i],TRUE,0,FALSE);
    ......
}

2.5 区分事件类型

#include<winsock2.h>

int WSAEnumNetworkEvents(
SOCKET s,                            //发生事件的套接字句柄
WSAEVENT hEventObject,               //与套接字相连的signaled状态的事件对象句柄
LPWSANETWORKEVENTS lpNetworkEvents   //保存发生的事件类型信息和错误信息的
                                     //WSANETWORKEVENTS结构体变量地址值
);
成功返回0
失败返回SOCKET_ERROR
struct _WSANETWORKEVENTS
{
    long lNetworkEvents;            //事件类型
    int iErrorCode[FD_MAX_EVENTS];  //错误信息
}WSANETWORKEVENTS,*LPWSANETWORKEVENTS;

事件类型的验证:

就是FD_READ、FD_ACCEPT等,和WSAEventSelect第三个参数一样。

错误信息的验证:

如果发生FD_XXX相关错误,则在iErrorCode[FD_XXX_BIT]中保存除0以外的其他值。

如:

WSANETWORKEVENTS netEvents;
......
WSAEnumNetworkEvents(hSock,hEvent,netEvents);
......
if(netEvents.lNetworkEvents & FD_ACCEPT)
{
    ......
}
......
if(netEvents.iErrorCode[FD_READ_BIT]!=0)
{
    ......
}

3.用异步通知I/O模型实现回声服务器端

#include<iostream>
#include<WinSock2.h>
#include<Windows.h>
#include<process.h>
#include<string>
#include<vector>

std::vector<SOCKET> vecSocket;
std::vector<WSAEVENT> vecEvent;

void ErrorHandle(WSANETWORKEVENTS network);

int main()
{
	WSADATA wsaData;
	if (0 != WSAStartup(MAKEWORD(2, 2), &wsaData))
	{
		std::cout << "start up fail!" << std::endl;
		return 0;
	}
	SOCKET server = socket(PF_INET, SOCK_STREAM, IPPROTO_TCP);
	if (server == INVALID_SOCKET)
	{
		std::cout << "socket fail!" << std::endl;
		return 0;
	}

	sockaddr_in serverAddr;
	memset(&serverAddr, 0, sizeof(serverAddr));
	serverAddr.sin_family = AF_INET;
	serverAddr.sin_addr.s_addr = htonl(INADDR_ANY);
	serverAddr.sin_port = htons(9130);

	if (SOCKET_ERROR == bind(server, (sockaddr*)&serverAddr, sizeof(serverAddr)))
	{
		std::cout << "bind fail!" << std::endl;
		return 0;
	}
	if (SOCKET_ERROR == listen(server, 2))
	{
		std::cout << "listen fail!" << std::endl;
		return 0;
	}
	
	WSAEVENT serverEvent=WSACreateEvent();
	if (SOCKET_ERROR == WSAEventSelect(server, serverEvent, FD_ACCEPT))
	{
		std::cout << "event select fail!" << std::endl;
		return 0;
	}

	vecSocket.push_back(server);
	vecEvent.push_back(serverEvent);

	while (1)
	{
		int eventSize = vecEvent.size();
		int res=WSAWaitForMultipleEvents(eventSize, vecEvent.data(), FALSE, WSA_INFINITE, FALSE);
		int a = (int)WSA_INVALID_EVENT;
		if (res == WSA_WAIT_FAILED)
		{
			std::cout << "wait fail!" << std::endl;
			//continue;
		}
		int first = res - WSA_WAIT_EVENT_0;
		for (int i = first; i < eventSize; ++i)
		{
			int sig = WSAWaitForMultipleEvents(1, &vecEvent[i], TRUE, 0, FALSE);
			if (sig == WSA_WAIT_FAILED)
				continue;
			int index = sig - WSA_WAIT_EVENT_0;
			WSANETWORKEVENTS network;
			int result = WSAEnumNetworkEvents(vecSocket[i], vecEvent[i], &network);
			if (result == SOCKET_ERROR)
			{
				ErrorHandle(network);
			}
			else
			{
				if (network.lNetworkEvents & FD_ACCEPT)
				{
					SOCKET client;
					sockaddr_in clientAddr;
					memset(&clientAddr, 0, sizeof(clientAddr));
					int clientAddrLen = sizeof(clientAddr);
					client=accept(vecSocket[i], (sockaddr*)&clientAddr, &clientAddrLen);
					if (INVALID_SOCKET==client)
					{
						std::cout << "accept fail!" << std::endl;
						continue;
					}
					else
					{
						WSAEVENT clientEvent = WSACreateEvent();
						WSAEventSelect(client, clientEvent, FD_READ|FD_CLOSE);

						vecSocket.push_back(client);
						vecEvent.push_back(clientEvent);
					}
				}
				else if (network.lNetworkEvents & FD_READ)
				{
					char buff[1024];
					int readLen=recv(vecSocket[i], buff, sizeof(buff), 0);
					std::cout << "客户端发来的消息:" << buff << std::endl;
					if (readLen != 0)
					{
						send(vecSocket[i], buff, readLen, 0);
					}
				}
				else if (network.lNetworkEvents & FD_CLOSE)
				{
					closesocket(vecSocket[i]);
					CloseHandle(vecEvent[i]);
					auto itSocket = vecSocket.begin() + i;
					if(itSocket<vecSocket.end())
						vecSocket.erase(itSocket);
					auto itEvent = vecEvent.begin() + i;
					if (itEvent < vecEvent.end())
						vecEvent.erase(itEvent);
				}
			}
		}
	}
	CloseHandle(serverEvent);
	closesocket(server);
	WSACleanup();
	return 0;
}

void ErrorHandle(WSANETWORKEVENTS network)
{
	if (network.iErrorCode[FD_ACCEPT_BIT]!=0)
	{
		std::cout << "accept error!" << std::endl;
	}
	else if (network.iErrorCode[FD_READ_BIT] != 0)
	{
		std::cout << "read error!" << std::endl;
	}
	else if (network.iErrorCode[FD_CLOSE_BIT] != 0)
	{
		std::cout << "close error!" << std::endl;
	}
}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
C++中的异步I/O允许程序在等待I/O操作完成时执行其他任务,从而提高程序的效率和性能。异步I/O可以通过以下两种方式实现: 1. 回调函数 在C++中,可以使用回调函数来实现异步I/O。当一个I/O操作完成时,操作系统会调用事先注册的回调函数,通知程序I/O操作已经完成,然后程序可以继续执行其他任务。回调函数通常在I/O操作发起时被注册,并且需要传入一个指向该函数的指针。 使用回调函数的一个例子是boost.asio库。该库提供了一套异步I/O的API,可以通过回调函数来处理异步I/O操作的完成。以下是一个简单的使用boost.asio库的异步读取文件的示例: ```c++ #include <iostream> #include <boost/asio.hpp> void read_handler(const boost::system::error_code& ec, std::size_t bytes_transferred) { if (!ec) { std::cout << "Read " << bytes_transferred << " bytes\n"; } } int main() { boost::asio::io_service io_service; boost::asio::posix::stream_descriptor descriptor(io_service, ::dup(STDIN_FILENO)); std::vector<char> buffer(1024); descriptor.async_read_some(boost::asio::buffer(buffer), read_handler); io_service.run(); return 0; } ``` 在上述示例中,异步读取文件的操作是通过`descriptor.async_read_some()`函数发起的,该函数会在I/O操作完成时调用`read_handler()`函数进行回调处理。 2. 异步操作对象 另一种实现异步I/O的方式是使用异步操作对象。异步操作对象是一个包含I/O操作和回调函数的对象,当I/O操作完成时,操作对象会自动调用回调函数进行回调处理。异步操作对象通常是通过某个I/O库或框架提供的。 使用异步操作对象的一个例子是Boost.Beast库中的`async_read_some()`函数。该函数会在I/O操作完成时自动调用回调函数进行回调处理。以下是一个简单的使用Boost.Beast库的异步读取HTTP消息的示例: ```c++ #include <iostream> #include <boost/asio.hpp> #include <boost/beast.hpp> void read_handler(const boost::system::error_code& ec, std::size_t bytes_transferred) { if (!ec) { std::cout << "Read " << bytes_transferred << " bytes\n"; } } int main() { boost::asio::io_service io_service; boost::asio::ip::tcp::socket socket(io_service); boost::beast::http::request<boost::beast::http::string_body> request; std::vector<boost::asio::mutable_buffer> buffers = request.prepare(boost::beast::http::max_request_size); socket.async_read_some(buffers, read_handler); io_service.run(); return 0; } ``` 在上述示例中,异步读取HTTP消息的操作是通过`socket.async_read_some()`函数发起的,该函数会在I/O操作完成时自动调用`read_handler()`函数进行回调处理。 总之,异步I/O是一种提高程序效率和性能的重要方式,可以通过回调函数或异步操作对象来实现。在C++中,可以使用boost.asio库、Boost.Beast库或其他I/O库来进行异步I/O编程。

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

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

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值