文档章节

vc下使用windows的性能计数器简介

rise-worlds
 rise-worlds
发布于 2016/06/20 13:41
字数 2403
阅读 1
收藏 0
 

/*   vc下使用windows的性能计数器简介    */
/*      作者:Rise */

/*
Microsoft Windwos NT/2000 提供了一个强大的API集来访问系统事件和性能数据的众多计数器。我们既可以实时地得到计数器的值,也可以从一个日志文件中读取计数器数据。功能可为强大,而且使用简单。
下面我就简单谈谈在vc中如何使用windows的性能计数器。好,废话少说,我们开始:

我们用一个简单的例子来说明性能计数器的使用方法。
比如:我们如何获取当前正在运行的某个进程的CPU使用率呢?你一定会说:“这还不简单,方法有很多”。当然,我承认这个不难,而且的确有很多方法。但是哪种方法最简单?效率最高呢?我猜大概
是使用性能计数器了。
*/

// 要使用性能计数器的基本步骤是:
// 1.打开计数器PdhOpenQuery;
// 2.为计数器句柄分配空间;
// 3.把感兴趣的计数器添加进来PdhAddCounter;
// 4.收集数据PdhCollectQueryData
// 4.得到计数器的数值PdhGetFormattedCounterValue;
// 5.关闭计数器PdhCloseQuery。

// 下面是用代码实现的步骤
// 第一步:
// 在头文件中 
#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的性能计数器可以获得好几百项系统计数信息,几乎所有和计数有关的信息都可以得到。说到这里一定有朋友要问:“我还能得到哪些信息?这么多的计数器又代表什么含义?”,我们继续向下看。
    上面说过了,要获取其它技术信息只需更改计数器地址(就是PdhAddCounter函数中的第二个参数“\Process(( "QQ" ))\%Processor Time”),每个计数器地址包含三个部分(计数器对象Process、计数器%Processor Time、计数器实例QQ),我们只要知道你的系统中都有哪些计数器对象、每个计数器对象有包含哪些计数器、每个计数器又有哪些计数器实例,按照上面的调用格式就可以得到你想要的所有计数信息。
    Microsoft为我们提供了方便获取计数器对象、计数器、实例信息的方法---杖举。
    要杖举计数器需要用到以下几个API:
    1.杖举计数器对象
    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
 
*/

// 杖举计数器对象的基本步骤是:
// 1.获取计数器对象列表大小
// 2.为计数器列表分配缓冲区
// 3.开始杖举

// 以下是编程实现:
// 第一步:获取计数器对象列表大小
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

/*
 通过刚才杖举得到计数器对象就可以继续杖举该对象下的计数器和计数器实例,方法和上面基本雷同,有兴趣的朋友可以自己来做,限于篇幅我就不重复了。
 我在说说如何知道计数器的描述信息(可是中文的哦!),也就是每个计数器都代表什么含义?干什么用的?要知道每个计数器描述信息需要用到PdhGetCounterInfo函数(都是在pdh开头的API中打转)。
*/

// 基本步骤如下:
// 1.格式化某一个计数器地址(字符串)
/*
 在这里需要说明一下:有很多计数器是没有实例的。有实例和没有实例的格式化形式略有不同。
 比如:
  (有实例的)获取当前写入操作时传送到磁盘上的字节速度:需要用到”PhysicalDisk“计数器对象、该计数器对象下的"Disk Write Bytes/sec"计数器、以及计数器实例(在我的机子上主硬盘的实例为 "0 C: D: E: F:" ) ,那么获取我传送到主硬盘上的字节速度的计数器地址为 : "\PhysicalDisk("0 C: D: E: F:")\Disk Write Bytes/sec" 。
  (无实例的)获取本计算机自上次启动后已经运行的时间(单位秒):需要用到"System"计数器对象、盖计数器对象下的"System Up Time"计数器、无实例,那么这个地址为: "\System\System Up Time" 。
// 2.创建计数器PdhAddCounter
// 3.分配接收描述信息的缓冲区
// 4.获取描述信息
*/

// 以下是程序实现:

__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编程经验的人肯定对此文嗤之以鼻,希望看在广大初学者的份上(包括我)请不要言语攻击我。
 希望有经验的朋友多提宝贵意见、多斧正。
 
 最后说明:
 1.本文采用的是UNICODE编码,其中用到了一些宏,如果要不加修改直接通过编译请在编译器中选择UNICODE编码,并在头文件中添加 #include <TChar.h> (略作修改就可在ANSI编码下运行)
 2.本文用到的__try {} __finally {} 只是结构化异常处理SEH,可以不要。
 3.需要本文例子工程及源代码的朋友请到我的主页上来下载 www.Clock.5888.com
 4.性能计数器只能用在2000/Xp系统(2003没试过)
 5.本文代码编译环境 Windows 2000 + VC.net
 6.欢迎转载,转载请注明文章作者和出处。 
*/

本文转载自:http://www.cnblogs.com/flying_bat/archive/2005/05/11/153219.html

共有 人打赏支持
rise-worlds

rise-worlds

粉丝 2
博文 1755
码字总数 0
作品 0
深圳
程序员
私信 提问
vc第三方界面库BCGControlBar与Xtreme Toolkit对比

vc第三方界面库BCGControlBar与Xtreme Toolkit对比 BCGControlBar简介 BCGControlBar专业版(BCGControlBar Pro for MFC )是MFC的扩展类库,一直是MFC界面领域的老大哥,而且是微软合作商,对...

baobiaokongjian
2014/08/16
0
0
系统性能指标查看方法-Windows

一、CPU使用率 1、Windows上最常用的CPU使用率监控工具是Task Manager(任务管理器)和Performance Monitor(性能监视器)。这两个监控工具用不同颜色区分用户态CPU使用率和系统态CPU使用率。...

K_ONE
2016/11/26
230
0
对SQLSERVER进行性能监控

原文:对SQLSERVER进行性能监控 对SQLSERVER进行性能监控 在上一篇文章《SQLSERVER性能监控级别步骤》里说到性能监控的步骤中有一步涉及到建立性能基线,但是没有说到有哪些计数器 可以用来进...

杰克.陈
2018/08/17
0
0
性能计数器与profiler的组合性能诊断

性能计数器和sql profiler都是常用的性能诊断工具和优化工具,最近和群友聊天发现很多人竟然不知道这两个可以“组合”使用,所以这篇算是一篇扫盲贴吧。 两种工具简述   通过计数器可以收集...

nothingfinal
2018/03/07
0
0
如何用Windows 性能监视器进行SMB性能监控和分析

SMB是Windows系统上主要的共享文件访问协议。作为微软设计的原生文件系统协议,和其它网络文件系统协议相比,SMB提供Windows文件系统语义,并且在网络短时间断开和服务器重启等出错情况下能保...

nas-hz
2018/09/30
0
0

没有更多内容

加载失败,请刷新页面

加载更多

组合问题(先提取数字+全组合)

今天在网上看到一个问题:想从A,B,C,D,E字母中选取3个A,B,C;并做出全组合ABC,ACB,BAC,BCA,CBA,CAB。这样的结果会有多少? 想法也是和问题一致: 1. 先从数列中选取所需要的值: A,B,C,D,E中选取...

tedzheng
33分钟前
1
0
vi常用命令

记录存档用,如下: 1、打开命令: vi+filename 2、退出命令: :q 退出而且不保存修改的内容 :q! 强制退出不保存修改的内容 :wq 退出并且保存修改的内容 :wq! 强制保存修改的内容然后退出...

ZICK_ZEON
34分钟前
3
0
查看Mysql正在执行的事务、锁、等待

一、关于锁的三张表(MEMORY引擎) ## 当前运行的所有事务mysql> select * from information_schema.innodb_trx\G;*************************** 1. row *************************** ......

吴伟祥
34分钟前
5
0
判断ifream 是否加载完成

$(function(){var iframe = document.getElementById("mainFrames"); if (iframe.attachEvent){ iframe.attachEvent("onload", function(){ //你要做的事}); }els......

卖星星的小矮人
37分钟前
2
0
11 Git —— 自定义Git

11 Git —— 自定义Git 忽略特殊文件 有些时候,你必须把某些文件放到Git工作目录中,但又不能提交它们,比如保存了数据库密码的配置文件啦,等等,每次git status都会显示Untracked files ....

lwenhao
44分钟前
3
0

没有更多内容

加载失败,请刷新页面

加载更多

返回顶部
顶部