CSDN博客

img hcj2002

在VC++通过汇编实现获取代码运行时间

发表于2004/6/13 16:10:00  2584人阅读

分类: Visual C++

                         在VC++通过汇编实现获取代码运行时间

     如何获得程序或者一段代码运行的时间?你可能说有专门的程序测试工具,确实,不过你也可以在程序中嵌入汇编代码来实现。
     在Pentium的指令系统中有一条指令可以获得CPU内部64位计数器的值,我们可以通过代码两次获取该计数器的值而获得程序或代码运行的时钟周期数,进而通

过你的cpu的频率算出一个时钟周期的时间,从而算出程序运行的确切时间。
    我们通过指令TDSIC来获得cpu内部计数器的值,指令TDSIC返回值放在EDX:EAX中,其中EDX中存放64位寄存器中高32位的值,EAX存放第32位的值.

下面看看实现的代码:
=============================================================================================
//用汇编实现获取一段代码运行的时间

#include<iostream>

using namespace std;

//void GetClockNumber (long high, long low);
//void GetRunTime();
 
int main()
{      
 
 long HighStart,LowStart,HighEnd,LowEnd;
 long numhigh,numlow;
 //获取代码运行开始时cpu内部计数器的值
 __asm             
 {
  RDTSC
  mov HighStart, edx
  mov LowStart, eax
 }
 for(int i= 0; i<100000; i++ )
 {
             for(int i= 0; i<100000; i++ )
      {
      
      }
 }
 //获取代码结束时cpu内部计数器的值,并减去初值
        __asm
 {
  //RDTSC
  //mov HighEnd, edx
  //Mov LowEnd,  eax
  //;获取两次计数器值得差
 //--------------------------------------------------
  //sub eax,  LowStart
  //cmp    eax,  0       ; 如果低32的差为负则求返,因为第二次取得永远比第一次的大
  //jg     L1
  //neg     eax
  //jmp     L2
  //          L1: mov numlow,  eax
  //          L2: sbb edx,  HighStart
  // mov numhigh, edx
  //--------------------------------------------------------
  //以前代码有逻辑错误,其实直接用sbb减就可以了,而不必考虑低32位是否高于高32
  RDTSC
  mov HighEnd, edx
  Mov LowEnd,  eax
  sub eax, LowStart
  mov     numlow,  eax
  mov     eax,edx
  sbb     eax,HighStart 
  mov     numhigh,eax

 }
        //把两个计数器值之差放在一个64位的整形变量中
        //先把高32位左移32位放在64的整形变量中,然后再加上低32位
         __int64  timer =0;

        timer |= numhigh;
        timer = timer<<32;
        timer |= numlow;  
      
        //输出代码段运行的时钟周期数
         //以频率1.1Gcpu为例,如果换计算机把其中的1.1改乘其它即可,因为相信大家的cpu都应该在1G以上  ^_^
 cout<< (double) (timer /1.1/1000000000) << endl;
 return 0;
}

     这样通过一条简单的汇编指令就可以获得程序或一段代码的大概时间,不过并不能得到运行的确切时间,因为即使去掉中间的循环,程序也会有个运行时间,

因为在第一次取得计数器的值后,有两条汇编指令mov HighStart, edx       mov LowStart, eax这两条指令当然也有运行时间 ,当然你可以减去这两条指令的运

行时间(在1.1G的机子上是3e-8s),这样会更精确一点。^_^
    如果你要确切知道程序的运行时间,专业的测试软件肯定会更好一点,不过好像一般没有必要获取除非专门的要求的程序。不过能DIY一个也是不错的,不管有

没有,最起码你可以学到在VC++中如何嵌入汇编代码以及如何使用32位的寄存器,其实和16位的寄存器一样使用,将来64的也应该一样,只不过位数不同罢了。
^_^

-----------------------------------------------------------------------------------------------
     发现程序有错误当第一次取得计数器的值为 0xFFFFFFFF时,第二次取0x00000000最后输出的值肯定是负值。对于某个cpu来说,可以计时的最大时间为0xFFFFFFFF/cpu频率(单位为G)/1000000000s。
     所以只能用来计算运行时间小于最大时间的代码才可能是正确的(也有可能是错误的)。
另外,如果是双cpu的话,会出现什么问题???如何才能解决上面提到的问题??
思考中……

感谢 IPLinger 提出程序中的问题,及提出自己的想法,才使我发现了其中的错误,在此表示感谢!
                                                                                                                       
                                                                                                                             2004年11月10日

 

0 0

相关博文

我的热门文章

img
取 消
img