在Windows系统里使用完成端口

简介:     在Windows系统里,使用完成端口高性能的方法之一,比如把完成端口使用到线程池和网络服务器里。
    在Windows系统里,使用完成端口高性能的方法之一,比如把完成端口使用到线程池和网络服务器里。现在就通过线程池的方法来介绍怎么样使用完成端口,高性能的服务器以后再仔细地介绍怎么样构造它。其实完成端口一个队列,所有的线程都在等消息出现,如果队列里有消息,就每个线程去获取一个消息执行它。先用函数CreateIoCompletionPort来创建一个消息队列,然后使用GetQueuedCompletionStatus函数来从队列获取消息,使用函数PostQueuedCompletionStatus来向队列里发送消息。通过这三个函数就实现完成端口的消息循环处理。


HANDLE CreateIoCompletionPort(
    HANDLE FileHandle,//关联的文件句柄
    HANDLE ExistingCompletionPort,//已经存在的完成端口
    ULONG_PTR CompletionKey,//传送给处理函数的参数
    DWORD NumberOfConcurrentThreads//有多少个线程在访问这个消息队列
    );


BOOL GetQueuedCompletionStatus(
    HANDLE CompletionPort,//已经存在的完成端口
    LPDWORD lpNumberOfBytesTransferred,
    PULONG_PTR lpCompletionKey,//传送给处理函数的参数
    LPOVERLAPPED *lpOverlapped,//传送给处理函数的参数
    DWORD dwMilliseconds//等待时间
    );


BOOL PostQueuedCompletionStatus(
    HANDLE CompletionPort,
    DWORD dwNumberOfBytesTransferred,//传送了多少个字节
    ULONG_PTR dwCompletionKey,
    LPOVERLAPPED lpOverlapped
    );


//调用函数的例子如下:
#pragma once
#include "Thread.h"


//使用IOCP实现线程池。
class CThreadPools
{
public:
CThreadPools(void)
{
m_nThreadCount = 2;
}


bool Init(void)
{
//创建一个IOCP。
m_hQueue = CreateIoCompletionPort(INVALID_HANDLE_VALUE, NULL, 0, m_nThreadCount);
if (m_hQueue == NULL)
{
//创建IOCP失败。
return false;
}
}


int GetThreadCount(void) const
{
return m_nThreadCount;
}


//线程池处理的内容。
DWORD Run(void)
{
DWORD dwBytesTransfered;
ULONG_PTR dwCompletionKey;
OVERLAPPED* pOverlapped;


//等一个IOCP的消息。
while (GetQueuedCompletionStatus(m_hQueue, &dwBytesTransfered, &dwCompletionKey, &pOverlapped, INFINITE))
{
if (pOverlapped == ((OVERLAPPED*)((__int64)-1)))
{
//退出。
OutputDebugString(_T("退出 /r/n"));
break;
}else{
WPARAM request = (WPARAM) dwCompletionKey;
//处理消息。
OutputDebugString(_T("GetQueuedCompletionStatus /r/n"));
}
}
return 0;
}


//发送处理的消息。
bool QueueRequest(WPARAM wParam)
{        
//往IOCP里发送一条消息。
if (!PostQueuedCompletionStatus(m_hQueue, 0, (ULONG_PTR) wParam, NULL))
return false;
return true;
}


//关闭所有线程。
void Close(void)
{
for (int i = 0; i < m_nThreadCount; i++)
PostQueuedCompletionStatus(m_hQueue, 0, 0, (OVERLAPPED*) ((__int64) -1) );
}


protected:
//接收消息处理的队列。
HANDLE m_hQueue;
//线程个数。
int m_nThreadCount;
};


class CThreads : public CThread
{
public:
CThreads(CThreadPools* pPool)
{
m_pPool = pPool;        
}


protected:
//线程运行函数。
//在这里可以使用类里的成员,也可以让派生类实现更强大的功能。
virtual DWORD Run(void)
{
if (m_pPool)
return m_pPool->Run();
return -1;
}


protected:
CThreadPools* m_pPool;
};



目录
相关文章
|
3月前
|
安全 Windows
永久关闭 Windows 11 系统更新
永久关闭 Windows 11 系统更新
168 0
|
2月前
|
安全 Windows
【Azure Cloud Service】在Windows系统中抓取网络包 ( 不需要另外安全抓包工具)
通常,在生产环境中,为了保证系统环境的安全和纯粹,是不建议安装其它软件或排查工具(如果可以安装,也是需要走审批流程)。 本文将介绍一种,不用安装Wireshark / tcpdump 等工具,使用Windows系统自带的 netsh trace 命令来获取网络包的步骤
73 32
|
2月前
|
存储 负载均衡 Java
如何配置Windows主机MPIO多路径访问存储系统
Windows主机多路径(MPIO)是一种技术,用于在客户端计算机上配置多个路径到存储设备,以提高数据访问的可靠性和性能。本文以Windows2012 R2版本为例介绍如何在客户端主机和存储系统配置多路径访问。
119 13
如何配置Windows主机MPIO多路径访问存储系统
|
3月前
|
Windows
Windows系统命令dir使用详解
Windows系统命令dir使用详解
186 2
|
4月前
|
Linux Windows
Windows系统批量创建文件夹的技巧
Windows系统批量创建文件夹的技巧
128 1
|
3月前
|
Windows
.NET 隐藏/自定义windows系统光标
【10月更文挑战第20天】在.NET中,可以使用`Cursor`类来控制光标。要隐藏光标,可将光标设置为`Cursors.None`。此外,还可以通过从文件或资源加载自定义光标来更改光标的样式。例如,在表单加载时设置`this.Cursor = Cursors.None`隐藏光标,或使用`Cursor.FromFile`方法加载自定义光标文件,也可以将光标文件添加到项目资源中并通过资源管理器加载。这些方法适用于整个表单或特定控件。
|
3月前
|
Apache 数据中心 Windows
将网站迁移到阿里云Windows系统云服务器,访问该站点提示连接被拒绝,如何处理?
将网站迁移到阿里云Windows系统云服务器,访问该站点提示连接被拒绝,如何处理?
|
3月前
|
域名解析 缓存 网络协议
Windows系统云服务器自定义域名解析导致网站无法访问怎么解决?
Windows系统云服务器自定义域名解析导致网站无法访问怎么解决?
|
3月前
|
运维 网络安全 虚拟化
Windows系统镜像检测修复建议
Windows系统镜像检测修复建议
|
4月前
|
监控 网络安全 Windows
Windows系统命令
Windows系统命令
37 1

热门文章

最新文章