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

Wesley13
• 阅读 638

/*   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.欢迎转载,转载请注明文章作者和出处。 
*/

点赞
收藏
评论区
推荐文章
blmius blmius
3年前
MySQL:[Err] 1292 - Incorrect datetime value: ‘0000-00-00 00:00:00‘ for column ‘CREATE_TIME‘ at row 1
文章目录问题用navicat导入数据时,报错:原因这是因为当前的MySQL不支持datetime为0的情况。解决修改sql\mode:sql\mode:SQLMode定义了MySQL应支持的SQL语法、数据校验等,这样可以更容易地在不同的环境中使用MySQL。全局s
皕杰报表之UUID
​在我们用皕杰报表工具设计填报报表时,如何在新增行里自动增加id呢?能新增整数排序id吗?目前可以在新增行里自动增加id,但只能用uuid函数增加UUID编码,不能新增整数排序id。uuid函数说明:获取一个UUID,可以在填报表中用来创建数据ID语法:uuid()或uuid(sep)参数说明:sep布尔值,生成的uuid中是否包含分隔符'',缺省为
捉虫大师 捉虫大师
3年前
实现一个比LongAdder更高性能的计数器有多难
本文已收录https://github.com/lkxiaolou/lkxiaolou欢迎star。强悍的LongAdderLongAdder是jdk8引入的适用于统计场景的线程安全的计数器。在此之前,实现一款线程安全的计数器要么加锁,要么使用AtomicLong,加锁性能必然很差,AtomicLong性能要好很多,但是在高并发、多线程下,也显得吃力。
Jacquelyn38 Jacquelyn38
3年前
2020年前端实用代码段,为你的工作保驾护航
有空的时候,自己总结了几个代码段,在开发中也经常使用,谢谢。1、使用解构获取json数据let jsonData  id: 1,status: "OK",data: 'a', 'b';let  id, status, data: number   jsonData;console.log(id, status, number )
DevOpSec DevOpSec
3年前
内存问题定位与解决
内存问题定位基本流程:主要用到的性能计数器1.Pagelifeexpectancy (数据库计数器:主要显示不被使用的页,将在缓存中停留的秒数)2.
Easter79 Easter79
3年前
Twitter的分布式自增ID算法snowflake (Java版)
概述分布式系统中,有一些需要使用全局唯一ID的场景,这种时候为了防止ID冲突可以使用36位的UUID,但是UUID有一些缺点,首先他相对比较长,另外UUID一般是无序的。有些时候我们希望能使用一种简单一些的ID,并且希望ID能够按照时间有序生成。而twitter的snowflake解决了这种需求,最初Twitter把存储系统从MySQL迁移
Wesley13 Wesley13
3年前
00:Java简单了解
浅谈Java之概述Java是SUN(StanfordUniversityNetwork),斯坦福大学网络公司)1995年推出的一门高级编程语言。Java是一种面向Internet的编程语言。随着Java技术在web方面的不断成熟,已经成为Web应用程序的首选开发语言。Java是简单易学,完全面向对象,安全可靠,与平台无关的编程语言。
Stella981 Stella981
3年前
Android蓝牙连接汽车OBD设备
//设备连接public class BluetoothConnect implements Runnable {    private static final UUID CONNECT_UUID  UUID.fromString("0000110100001000800000805F9B34FB");
Stella981 Stella981
3年前
JVM垃圾回收算法
一、如何判断对象时候需要回收1.引用计数法        给对象添加一个引用计数器,每当有一个地方引用它,计数器加1;引用失效时,计数器减1。计数器为0的对象就表示不可用。      优点:效率高,实现简单。      缺点:对象间如果存在循环引用的情况,就会导致计数器不可能为0,计数器无法通知GC进行回收。2.可达性分析算法
Stella981 Stella981
3年前
JVM垃圾回收机制
引用计数法:当给对象添加一个引用计数器,每当有一个地方引用这个对象时计数器值就1;引用失效时,计数器值就1;任何时刻计数器为0的对象就是不可能在被使用。优点:引用计数收集器可以很快地执行,交织在程序运行中。缺点:无法检测出循环引用。例如:MyObjectobject1newMyObject();MyObjectobject2