分类: C/C++
2008-04-23 21:57:49
VC下使用 Windows 的性能计数器简介
作者:
前言
Microsoft Windwos NT/2000 提供了一个强大的API集来访问系统事件和性能数据的众多计数器。我们既可以实时地得到计数器的值,也可以从一个日志文件中读取计数器数据。功能可为强大,而且使用简单
。下面我就简单谈谈在VC中如何使用Windows的性能计数器。好,废话少说,我们开始。
我们用一个简单的例子来说明性能计数器的使用方法。
比如:我们如何获取当前正在运行的某个进程的CPU使用率呢?你一定会说:“这还不简单,方法有很多”。当然,我承认这个不难,而且的确有很多方法。但是哪种方法最简单?效率最高呢?我猜大概是使用性能计数器了。
要使用性能计数器的基本步骤是:
下面是用代码实现的步骤:
第一步:
在头文件中
#include <"Pdh.h">在实现文件中,
#pragma comment ( lib , "Pdh.lib" )第二步:打开计数器,并给计数器句柄分配空间
HQUERY hQuery = NULL; PDH_STATUS pdhStatus; HCOUNTER * pCounterHandle = NULL; __try { // 打开计数器 pdhStatus = PdhOpenQuery ( 0 , 0 , & hQuery ); if ( pdhStatus != ERROR_SUCCESS ) { __leave; } pCounterHandle = ( HCOUNTER * ) GlobalAlloc ( GPTR , sizeof ( HCOUNTER ) ); if ( pCounterHandle == NULL ) { __leave; } } __finally { if ( AbnormalTermination () ) { } }第三步:创建计数器(假设要获取QQ程序的CPU使用率)
PDH_FMT_COUNTERVALUE fmtValue ; DWORD dwctrType ; __try { pdhStatus = PdhAddCounter ( hQuery , _TEXT ( "Process(_TEXT ( "QQ" ))\\%Processor Time" ), 0, pCounterHandle ) ; if ( pdhStatus != ERROR_SUCCESS ) { __leave ; } pdhStatus = PdhCollectQueryData ( hQuery ) ; if ( pdhStatus != ERROR_SUCCESS ) { __leave ; } // 得到当前计数器值 pdhStatus = PdhGetFormattedCounterValue ( * pCounterHandle, PDH_FMT_DOUBLE, & dwctrType, & fmtValue ) ; if ( pdhStatus != ERROR_SUCCESS ) { __leave ; } // fmtValue.doubleValue就是当前此时此刻该程序的CPU使用率(循环调用就可得到实时数据) } __finally { if ( AbnormalTermination () ) { } }第四步:关闭计数器
pdhStatus = PdhCloseQuery ( hQuery ) ; if ( pdhStatus == ERROR_SUCCESS ) { // 关闭成功 } else { // 关闭失败 }是不是很简单呀!上面例子中PdhAddCounter函数是添加计数器,它的第二个参数就是计数器地址,我们可以更换其它的,以获得其它计数数据。(详细请查询MSDN)Windows 的性能计数器可以获得好几百项系统计数信息,几乎所有和计数有关的信息都可以得到。说到这里一定有朋友要问:“我还能得到哪些信息?这么多的计数器又代表什么含义?”,我们继续向下看。
PdhEnumObjects ( NULL , // [IN]数据源,NT4.0必须为NULL szMachineName , // [IN]机器名。本地机器为NULL szObjectListBuffer , // [OUT]接收计数器列表的缓冲区,如果计数器列表长度为0,则该项为空 & dwObjectListSize , // [IN/OUT]设置或接收计数器列表长度 dwDetailLevel , // 获取信息的级别 // PERF_DETAIL_NOVICE 初级级别 // PERF_DETAIL_ADVANCE 高级级别(包含初级) // PERF_DETAIL_EXPERT 专家级别(包含初级和高级) // PERF_DETAIL_WIZARD 系统级别(包含所有级别) TRUE );
2、杖举计数器和计数器实例
PdhEnumObjectItems ( NULL , // [IN]数据源,NT4.0必须为NULL szMachineName , // [IN]机器名。本地机器为NULL pctCounter , // [IN]计数器名 szCounterListBuffer , // [OUT]接收计数器列表的缓冲区,如果计数器列表长度为0,则该项为空 & dwCounterListSize , // [IN/OUT]设置或接收计数器列表长度 szInstanceListBuffer , // [OUT]接收实例列表的缓冲区,如果计数器列表长度为0,则该项为空 & dwInstanceListSize , // [IN/OUT]设置或接收实例列表长度 dwDetailLevel , // 获取信息的级别 // PERF_DETAIL_NOVICE 初级级别 // PERF_DETAIL_ADVANCE 高级级别(包含初级) // PERF_DETAIL_EXPERT 专家级别(包含初级和高级) // PERF_DETAIL_WIZARD 系统级别(包含所有级别) 0 ); // 最后一个参数系统保留为0
更详细信息请参阅MSDN
枚举计数器对象的基本步骤是:
以下是编程实现:
第一步:获取计数器对象列表大小
LPTSTR szObjectListBuffer = NULL ; DWORD dwObjectListSize = 0 ; LPTSTR szThisObject = NULL ; __try { // 第一次调用该函数获得接收性能计数器对象列表的缓冲区大小 pdhStatus = PdhEnumObjects ( NULL , // [IN]数据源,NT4.0必须为NULL NULL , // [IN]机器名。本地机器为NULL szObjectListBuffer , // [OUT]接收计数器列表的缓冲区,如果计数器列表长度为0,则该项为空 & dwObjectListSize , // [IN/OUT]设置或接收计数器列表长度 PERF_DETAIL_WIZARD , // 获取信息的级别 // PERF_DETAIL_NOVICE 初级级别 // PERF_DETAIL_ADVANCE 高级级别(包含初级) // PERF_DETAIL_EXPERT 专家级别(包含初级和高级) // PERF_DETAIL_WIZARD 系统级别(包含所有级别) true ) ; if ( pdhStatus != ERROR_SUCCESS ) { __leave ; } // 根据得到的缓冲区大小分配计数器对象列表缓冲区内存 szObjectListBuffer = ( LPTSTR ) malloc ( ( dwObjectListSize * sizeof ( TCHAR ) ) ) ; if ( szObjectListBuffer == NULL ) { __leave ; } // 第二次调用该函数获得计数器对象 pdhStatus = PdhEnumObjects ( NULL , // [IN]数据源,NT4.0必须为NULL NULL , // [IN]机器名。本地机器为NULL szObjectListBuffer , // [OUT]接收计数器列表的缓冲区,如果计数器列表长度为0,则该项为空 & dwObjectListSize , // [IN/OUT]设置或接收计数器列表长度 PERF_DETAIL_WIZARD , // 获取信息的级别 // PERF_DETAIL_NOVICE 初级级别 // PERF_DETAIL_ADVANCE 高级级别(包含初级) // PERF_DETAIL_EXPERT 专家级别(包含初级和高级) // PERF_DETAIL_WIZARD 系统级别(包含所有级别) TRUE ) ; if ( pdhStatus != ERROR_SUCCESS ) { __leave ; } szThisObject = szObjectListBuffer ; // 开始杖举 for ( ; * szThisObject != 0 ; szThisObject = ( lstrlen ( szThisObject ) 1 ) ) { // 每循环一次 szThisObject 就是杖举到的计数器对象 } } __finally { if ( AbnormalTermination () ) { // 如果失败 if ( szObjectListBuffer != NULL ) { free ( szObjectListBuffer ) ; szObjectListBuffer = NULL ; } } else { // 如果成功 } }最后别忘了 free
__try { // 创建计数器 pdhStatus = PdhAddCounter ( hQuery, _TEXT ( "\\System\\System Up Time" ), 0,pCounterHandle ); if ( pdhStatus != ERROR_SUCCESS ) { __leave ; } // 分配接收描述信息的缓冲区 DWORD dwCounterBuff ; BYTE byCounterBuff [ sizeof ( PDH_COUNTER_INFO ) sizeof ( TCHAR ) * 2048 ]; dwCounterBuff = sizeof ( byCounterBuff ); // 获取描述信息 pdhStatus = PdhGetCounterInfo ( * pCounterHandle, TRUE, & dwCounterBuff, ( PPDH_COUNTER_INFO ) byCounterBuff ); if ( pdhStatus != ERROR_SUCCESS ) { __leave ; } PDH_COUNTER_INFO pdhCounterInfo = * ( PPDH_COUNTER_INFO ) byCounterBuff; // 有关PDH_COUNTER_INFO结构的信息请参阅MSDN // PDH_COUNTER_INFO结构中包含了很多关于计数器的信息,其中szExplainText为计数器描述信息 // pdhCounterInfo.szExplainText } __finally { if ( AbnormalTermination () ) { // 如果失败 } else { // 如果成功 } }至此,关于性能计数器的简单介绍到此完毕。前面啰里啰唆说一大串,主要是考虑到刚接触VC不久的朋友,如果本文能对他们有帮助我将不胜荣幸。略有VC编程经验的人肯定对此文嗤之以鼻,希望看在广大初学者的份上(包括我)请不要言语攻击我。希望有经验的朋友多提宝贵意见、多斧正。