【windows核心编程】IO完成端口(IOCP)复制文件小例前简单说明
1、关于IOCP
IOCP即IO完成端口,是一种高伸缩高效率的异步IO方式,一个设备或文件与一个IO完成端口相关联,当文件或设备的异步IO操作完成的时候,去IO完成端口的【完成队列】取一项,根据完成键(Complete Key)来判断是哪个设备或文件的操作完成,然后再根据实际情况进行处理。
2、相关API 和 数据结构
将一个已完成的IO通知追加到IOCP的【完成队列】中
BOOL PostQueuedCompletionStatus(
HANDLE hCompletionPort, //IO完成端口
DWORD dwNumBytes, //传输的字节数
ULONG_PTR completionKey, //完成键
OVERLAPPED* pOverlapped //设备或文件发送异步IO请求时的关联OVERLAPPED结构指针
)
从【完成队列】中取一项
BOOL GetWQueuedCompletionStatus(
HANDLE hCompletionPort, //IOCP
PWORD pdwNumberOfBytesTransferred, //已传输字节数
PULONG_PTR pCompletionKey, //完成键
OVERLAPPED** ppOverlapped, //与文件或设备关联的OVERLAPPED结构,指针的指针,[out]
DWORD dwMilliseconds //等待时间
)
创建IOCP 或 将一个文件/设备与一个IOCP关联起来
HANDLE WINAPI CreateIoCompletionPort(
__in HANDLE FileHandle, //文件 设备句柄
__in_opt HANDLE ExistingCompletionPort, //一个IOCP句柄
__in ULONG_PTR CompletionKey, //完成键
__in DWORD NumberOfConcurrentThreads //允许运行的最大线程数量
);
新建
HANDLE hIOCP = CreateIoCompletionPort(INVALID_HANDLE_VALUE, NULL, 0, 一个数量);
关联
CreateIoCompletionPort(hFile, hIOCP, CK_XXX, 一个数量);
OVERLAPPED结构
typedef struct _OVERLAPPED {
ULONG_PTR Internal; //错误代码,系统管理
ULONG_PTR InternalHigh; //传输自字节数 系统管理
union {
struct {
DWORD Offset; //起始偏移 地位, 需要特别注意:大文件时这个值容易溢出
DWORD OffsetHigh; //起始偏移 高位
} ;
PVOID Pointer;
} ;
HANDLE hEvent; //事件内核对象
} OVERLAPPED, *LPOVERLAPPED;
IOCP的五个数据结构
①设备列表
表示与该端口相关联的一个或多个设备
②IO完成队列 FIFO
包含已传输字节数、完成键、OVERLAPPED结构、错误代码等
当IO请求完成 或 调用PostQueuedCompletionStatus时 会往该队列中添加项
③等待线程队列 FILO
当线程池中每个线程调用GetQueuedCompletionStatus时,调用线程的线程标示符会被添加到这个等待线程队列,这使得IOCP始终都能够知道有哪些线程正在等待对已完成的IO请求进行处理。
④已释放线程列表
对IO请求后进行处理的线程由于某种原因被挂起后,又被唤醒,就进入以释放线程列表
⑤已暂停线程列表
对IO请求后进行处理的线程由于某种原因被挂起后进入已暂停线程列表
3、说明
IOCP中运行运行的最大线程数一般是CPU个数
线程池中的最大线程数一般是CPU个数的2倍
下一篇用一个复制文件的例子来演示一下IOCP的使用
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· AI与.NET技术实操系列:基于图像分类模型对图像进行分类
· go语言实现终端里的倒计时
· 如何编写易于单元测试的代码
· 10年+ .NET Coder 心语,封装的思维:从隐藏、稳定开始理解其本质意义
· .NET Core 中如何实现缓存的预热?
· 25岁的心里话
· 闲置电脑爆改个人服务器(超详细) #公网映射 #Vmware虚拟网络编辑器
· 零经验选手,Compose 一天开发一款小游戏!
· 通过 API 将Deepseek响应流式内容输出到前端
· AI Agent开发,如何调用三方的API Function,是通过提示词来发起调用的吗