`

完成端口的原理

    博客分类:
  • VC
阅读更多

完成端口(I/O completion):
异步过程调用(apcs)问题:
    只有发overlapped请求的线程才可以提供callback函数(需要一个特定的线程为一个特定的I/O请求服务)。
 
完成端口(I/O completion)的优点:
    不会限制handle个数,可处理成千上万个连接。I/O completion port允许一个线程将一个请求暂时保存下来,由另一个线程为它做实际服务。
 
并发模型与线程池:
    在典型的并发模型中,服务器为每一个客户端创建一个线程,如果很多客户同时请求,则这些线程都是运行的,那么CPU就要一个个切换,CPU花费了更多的时间在线程切换,线程确没得到很多CPU时间。到底应该创建多少个线程比较合适呢,微软件帮助文档上讲应该是2*CPU个。但理想条件下最好线程不要切换,而又能象线程池一样,重复利用。I/O完成端口就是使用了线程池。
 
理解与使用:
第一步:
在我们使用完成端口之前,要调用CreateIoCompletionPort函数先创建完成端口对象。
定义如下:
HANDLE CreateIoCompletionPort(
                                HANDLE FileHandle,
                                HANDLE ExistingCompletionPort,
                                DWORD CompletionKey,
                                DWORD NumberOfConcurrentThreads
);

FileHandle:
文件或设备的handle, 如果值为INVALID_HANDLE_VALUE则产生一个没有和任何文件handle有关系的port.( 可以用来和完成端口联系的各种句柄,文件,套接字)
ExistingCompletionPort:
NULL时生成一个新port, 否则handle会加到此port上。
CompletionKey:
用户自定义数值,被交给服务的线程。GetQueuedCompletionStatus函数时我们可以完全得到我们在此联系函数中的完成键(申请的内存块)。在GetQueuedCompletionStatus
中可以完封不动的得到这个内存块,并且使用它。
 
NumberOfConcurrentThreads:
参数NumberOfConcurrentThreads用来指定在一个完成端口上可以并发的线程数量。理想的情况是,一个处理器上只运行一个线程,这样可以避免线程上下文切换的开销。如果这个参数的值为0,那就是告诉系统线程数与处理器数相同。我们可以用下面的代码来创建I/O完成端口。
 
隐藏在之创建完成端口的秘密:
1. 创建一个完成端口
CreateIoCompletionPort(INVALID_HANDLE_VALUE, 0, 0, dwNumberOfConcurrentThreads);
 
2. 设备列表,完成端口把它同一个或多个设备相关联。
CreateIoCompletionPort(hDevice, hCompPort, dwCompKey, 0) ;
 
第二步:
根据处理器个数,创建cpu*2个工作线程:
CreateThread(NULL, 0, ServerWorkerThread, CompletionPort,0, &ThreadID))
与此同时,服务器调用WSASocket,bind, listen, WSAAccept,之后,调用
CreateIoCompletionPort((HANDLE) Accept, CompletionPort... )把一个套接字句柄和一个完成端口绑定到一起。完成端口又同一个或多个设备相关联着,所以以套接字为基础,投递发送和请求,对I/O处理。接着,可以依赖完成端口,接收有关I/O操作完成情况的通知。再看程序里:
WSARecv(Accept, &(PerIoData->DataBuf), 1, &RecvBytes, &Flags,
 &(PerIoData->Overlapped), NULL)开始调用,这里象前面讲过的一样,既然是异步I/O,所以WSASend和WSARecv的调用会立即返回。
 
系统处理:
当一个设备的异步I/O请求完成之后,系统会检查该设备是否关联了一个完成端口,如果是,系统就向该完成端口的I/O完成队列中加入完成的I/O请求列。
 
然后我们需要从这个完成队列中,取出调用后的结果(需要通过一个Overlapped结构来接收调用的结果)。怎么知道这个队列中已经有处理后的结果呢,调用GetQueuedCompletionStatus函数。
 
工作线程与完成端口:
和异步过程调用不同(在一个Overlapped I/O完成之后,系统调用该回调函数。OS在有信号状态下(设备句柄),才会调用回调函数(可能有很多APCS等待处理了))
 
GetQueuedCompletionStatus
在工作线程内调用GetQueuedCompletionStatus函数。
GetQueuedCompletionStatus(
    HANDLE CompletionPort,
    LPDWORD lpNumberOfBytesTransferred,
    LPDWORD lpCompletionKey,
    LPOVERLAPPED *lpOverlapped,
    DWORD dwMilliseconds
);
CompletionPort:指出了线程要监视哪一个完成端口。很多服务应用程序只是使用一个I/O完成端口,所有的I/O请求完成以后的通知都将发给该端口。
lpNumberOfBytesTransferred:传输的数据字节数
lpCompletionKey:
完成端口的单句柄数据指针,这个指针将可以得到我们在CreateIoCompletionPort中申请那片内存。
lpOverlapped:
重叠I/O请求结构,这个结构同样是指向我们在重叠请求时所申请的内存块,同时和lpCompletionKey,一样我们也可以利用这个内存块来存储我们要保存的任意数据。
dwMilliseconds:
等待的最长时间(毫秒),如果超时,lpOverlapped被设为NULL,函数返回False.
 
GetQueuedCompletionStatus功能及隐藏的秘密:
GetQueuedCompletionStatus使调用线程挂起,直到指定的端口的I/O完成队列中出现了一项或直到超时。(I/0完成队列中出现了记录)调用GetQueuedCompletionStatus时,调用线程的ID(cpu*2个线程,每个ServerWorkerThread的线程ID)就被放入该等待线程队列中。
     等待线程队列很简单,只是保存了这些线程的ID。完成端口会按照后进先出的原则将一个线程队列的ID放入到释放线程列表中。
这样,I/O完成端口内核对象就知道哪些线程正在等待处理完成的I/O请求。当端口的I/O完成队列出现一项时,完成端口就唤醒(睡眠状态中变为可调度状态)等待线程队列中的一个线程。线程将得到完成I/O项中的信息:传输的字节数,完成键(单句柄数据结构)和Overlapped结构地址,线程是通过GetQueuedCompletionStatus返回这些信息,等待CPU的调度。
GetQueuedCompletionStatus返回可能有多种原因,如果传递无效完成端口句柄,函数返回False,GetLastError返回一个错误(ERROR_INVALID_HANDLE),如果超时,返回False, GetLastError返回WAIT_TIMEOUT, i/o完成队列删除一项,该表项是一个成功完成的I/O请求,则返回True。
 
    调用GetQueuedCompletionStatus的线程是后进先出的方式唤醒的,比如有4个线程等待,如果有一个I/O,最后一个调用GetQueuedCompletionStatus的线程被唤醒来处理。处理完之后,再调用GetQueuedCompletionStatus进入等待线程队列中。
 
深入分析完成端口线程池调度原理:
    假设我们运行在2CPU的机器上。创建完成端口时指定2个并发,创建了4个工作线程加入线程池中等待完成I/O请求,且完成端口队列(先入先出)中有3个完成I/O的请求的情况:
 工作线程运行, 创建了4个工作线程,调用GetQueuedCompletionStatus时,该调用线程就进入了睡眠状态,假设这个时候,I/O完成队列出现了三项,调用线程的ID就被放入该等待线程队列中。
 
 I/O完成端口内核对象(第3个参数等级线程队列),因此知道哪些线程正在等待处理完成的I/O请求。当端口的I/O完成队列出现一项时,完成端口就唤醒(睡眠状态中变为可调度状态)等待线程队列中的一个线程(前面讲过等待线程队列是后进先出)。所以线程D将得到完成I/O项中的信息:传输的字节数,完成键(单句柄数据结构)和Overlapped结构地址,线程是通过GetQueuedCompletionStatus返回这些信息。
在前面我们指定了并发线程的数目是2,所以I/O完成端口唤醒2个线程,线程D和线程C,另两个继续休眠(线程B,线程A),直到线程D处理完了,发现表项里还有要处理的,就唤醒同一线程继续处理。
 
线程并发量:
    并发量限制了与该完成端口相关联的可运行线程的数目, 它类似阀门的作用。 当与该完成端口相关联的可运行线程的总数目达到了该并发量,系统就会阻塞任何与该完成端口相关联的后续线程的执行, 直到与该完成端口相关联的可运行线程数目下降到小于该并发量为止。所以解释了线程池中的运行线程可能会比设置的并发线程多的原因。
    它的作用:
最有效的假想是发生在有完成包在队列中等待,而没有等待被满足,因为此时完成端口达到了其并发量的极限。此时,一个正在运行中的线程调用 GetQueuedCompletionStatus时,它就会立刻从队列中取走该完成包。这样就不存在着环境的切换,因为该处于运行中的线程就会连续不断地从队列中取走完成包,而其他的线程就不能运行了
注意:如果池中的所有线程都在忙,客户请求就可能拒绝,所以要适当调整这个参数,获得最佳性能。
线程并发:D线程挂起,加入暂停线程,醒来后又加入释放线程队列。
线程的安全退出:
PostQueudCompletionStatus函数,我们可以用它发送一个自定义的包含了OVERLAPPED成员变量的结构地址,里面包含一个状态变量,当状态变量为退出标志时,线程就执行清除动作然后退出。
 
完成端口使用需要注意的地方:
 1.在执行wsasend和wsarecv操作前,请先将overlapped结构体使用memset进行清零。

分享到:
评论

相关推荐

    IOCP完成端口原理

    IOCP 完成端口原理 一、IOCP 简介 IOCP(I/O Completion Port)是一种异步I/O模型,用于提高服务器的吞吐量和可扩展性。在Windows平台上,IOCP是实现高性能服务器的首选技术。IOCP的基本架构图包括完成端口、等待...

    IOCP完成端口原理[参照].pdf

    【IOCP完成端口原理】是Windows平台下用于高并发网络服务开发的关键技术,它能够高效地处理大量并发I/O操作。完成端口(IOCP)是一种先进先出(FIFO)队列,由操作系统的I/O子系统管理,当I/O操作完成后,操作系统会...

    Socket服务端完成端口.rar

    1. **完成端口原理**: 完成端口是一种异步I/O模型,通过将I/O操作与完成端口关联,系统会在I/O操作完成后自动将结果放入完成队列。服务端不再需要轮询检查I/O状态,而是通过注册回调函数或者线程池中的线程处理...

    完成端口使用说明及多个例子

    在本文中,我们将深入探讨完成端口的工作原理、创建与使用方法,并通过几个实际的例子来阐述如何在聊天室等服务器场景中应用完成端口。 一、完成端口工作原理 1. 异步I/O:完成端口的核心在于异步I/O操作,这意味...

    完成端口(vc完成端口下载文件的代码)

    在本文中,我们将深入探讨完成端口的工作原理,以及如何利用VC++(Visual Studio 2015)来实现一个基于完成端口的文件下载系统。 首先,理解完成端口的工作机制是至关重要的。完成端口通过将I/O操作与执行上下文...

    Delphi完成端口控件

    【Delphi完成端口控件】是一种在Delphi编程环境中实现高效并发I/O操作的组件。在Windows系统中,完成端口(IO ...通过合理地使用和理解完成端口的工作原理,开发者可以创建出更加稳定和高效的服务器端解决方案。

    完成端口服务端demo

    这个示例对于初学者理解完成端口工作原理和实践应用非常有价值。通过分析和学习这个代码,你可以更深入地理解如何在实际项目中应用完成端口技术,提高服务端的并发处理能力。同时,作者也欢迎对代码的不足之处进行...

    基于完成端口的服务器程序源代码

    首先,理解完成端口的工作原理至关重要。完成端口是一种I/O调度机制,它允许多个线程在一个或多个I/O操作完成后得到通知,并有效地分发这些操作到等待处理的线程。这种模型相对于传统的阻塞I/O模型,能显著提升...

    完成端口服务器简单示例

    下面将详细介绍完成端口的工作原理以及如何在VC++环境中实现这一模型。 完成端口是Windows系统提供的一种I/O机制,它允许应用程序通过一个单一的线程池处理来自多个句柄(如套接字、文件句柄等)的完成事件。这种...

    完成端口Console源码

    本篇文章将深入探讨完成端口的工作原理、创建与使用过程,以及如何通过提供的Console源码来学习和理解这一技术。 一、完成端口的工作原理 完成端口的核心思想是将I/O操作的完成通知与实际的处理工作解耦。当一个I/...

    《完成端口(CompletionPort)详解》源码

    《完成端口(Completion Port, 简称IOCP)详解》源码是关于网络编程中一种高效异步I/O模型的技术实现。IOCP,即完成端口,是Windows操作系统提供的一种多线程并行处理I/O操作的机制,特别适用于高并发、低延迟的服务器...

    一个简单的完成端口(服务端客户端)类源代码

    本文将深入探讨完成端口的工作原理、创建与使用,以及如何通过源代码实现一个简单的服务端和客户端。 一、完成端口的工作原理 完成端口基于异步I/O模型,它将I/O操作的调度与执行分离,从而避免了线程上下文切换的...

    Winsock 完成端口模型简介

    本文将深入探讨Winsock完成端口模型的工作原理、创建与使用,以及其在实际开发中的优势。 一、Winsock完成端口模型概述 Winsock 完成端口模型是基于异步I/O模型,允许应用程序在发起I/O操作后立即返回,无需等待...

    c++完成端口使用实例

    下面将详细阐述完成端口的工作原理及其在C++中的实现。 首先,完成端口是一种多路复用技术,它将多个套接字连接的I/O操作绑定到一个单一的线程池,由操作系统负责调度,将完成的I/O请求分发给线程处理。这种机制...

    网络模型之完成端口

    完成端口(Completion Port,简称IOCP)是Windows操作系统中的一种高效、多线程的I/O模型,主要用于处理大量的并发I/O操作。这种模型通过将I/O操作与完成线程池解耦,使得系统能够更好地管理和调度资源,提高系统的...

    完成端口例子--改进

    首先,完成端口的基本原理是将I/O操作与线程解耦。当一个I/O操作完成时,操作系统会将结果放入一个完成端口,等待关联的工作线程去处理。这种方式避免了线程阻塞等待I/O操作完成,从而提高了系统的并行处理能力。 ...

    基于完成端口模型的TCP通讯.rar_3EP_Sockets_T91_完成端口模型_网络通讯 UDP

    标题中的“基于完成端口模型的TCP通讯.rar_3EP_Sockets_T91_完成端口模型_网络通讯 UDP”表明这是一个关于使用完成端口(IO Completion Ports, I/OCP)模型实现TCP通信的示例项目,同时也涉及到UDP网络通信。...

    使用完成端口的例子(管理海量套接字)

    下面将详细阐述完成端口的原理、工作方式以及如何应用于套接字管理。 首先,了解完成端口的基本概念。完成端口是一个系统对象,用于集中处理多个I/O请求。当一个I/O操作完成后,系统会将结果放入完成端口,并通知...

    完成端口模式delphi源码

    首先,让我们理解完成端口的工作原理。完成端口是一种系统级对象,它可以关联到多个I/O操作,如网络套接字。当一个I/O操作完成时,系统会将结果放入完成端口,等待关联的线程来处理。这种方式允许单个或少数几个线程...

Global site tag (gtag.js) - Google Analytics