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

2016-02-19 14:25 40 1 收藏

下面图老师小编要跟大家分享VC++通过汇编获取代码运行时间,简单的过程中其实暗藏玄机,还是要细心学习,喜欢还请记得收藏哦!

【 tulaoshi.com - 编程语言 】

     如何获得程序或者一段代码运行的时间?你可能说有专门的程序测试工具,确实,不过你也可以在程序中嵌入汇编代码来实现。

(本文来源于图老师网站,更多请访问https://www.tulaoshi.com/bianchengyuyan/)

     在Pentium的指令系统中有一条指令可以获得CPU内部64位计数器的值,我们可以通过代码两次获取该计数器的值而获得程序或代码运行的时钟周期数,进而通过你的cpu的频率算出一个时钟周期的时间,从而算出程序运行的确切时间。

    我们通过指令TDSIC来获得cpu内部计数器的值,指令TDSIC返回值放在EDX:EAX中,其中EDX中存放64位寄存器中高32位的值,EAX存放第32位的值.

    下面看看实现的代码:


 //用汇编实现获取一段代码运行的时间
#includeiostream

using namespace std;

(本文来源于图老师网站,更多请访问https://www.tulaoshi.com/bianchengyuyan/)

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; i100000; i++ )
 {
             for(int i= 0; i100000; 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
 
 }
        //把两个计数器值之差放在一个64位的整形变量中
        //先把高32位左移32位放在64的整形变量中,然后再加上低32位
 __int64  timer =(numhigh32) + 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的也应该一样,只不过位数不同罢了。

来源:https://www.tulaoshi.com/n/20160219/1606942.html

延伸阅读
  Public Class Resize     '移动了吗?     Private IsMoving As Boolean = False     '控件最后的宽     Private ctrlLastWidth As Integer = 0     '控件最后的高     Private ctrlLastHeight As Integer =...
访问自定义属性 当属性与程序元素相关联后,可以使用反射来查询它们是否存在以及它们的值。用于查询属性的主要反射方法包含在 System.Reflection.MemberInfo.GetCustomAttributes 和 System.Reflection.Assembly.GetCustomAttributes 中。 自定义属性的可访问性根据附加该属性的程序集来进行检查。这相当于检查附加自定义属性的程序集中的类型...
6.5 客户机程序4在运行时获取连接参数 现在我们有了容易修改的防止出现错误的连接代码,我们要了解一些如何做某些比使用NULL 连接参数更灵巧的事情,如在运行时允许用户指定一些值。客户机程序3由于固定连接参数方面的缺陷,要想更改那些值中的任何一个,都必须编辑源文件并重新编译。这十分不方便,特别是想使程序用于其他人时。在...
飞信登录报C++运行时错误 故障现象: 登录时出现报错 原因分析: 原来安装过飞信,系统当中有之前的飞信文件存在。 解决方案: 在我的文档那个文件夹里,把那个叫fetion的文件夹删掉就可以了。
在VC中通过HTTP方式获取网页,我试过好几种方式,但是都有问题,现将我用的两种虽然也有问题但基本可以实现的方式写出来,如果大家有更好的方式或者能解决我所遇到的问题,请告诉我:QQ158792331,邮箱:werther161@yahoo.com.cn 第一种: 首先要包含这个头文件:#include afxinet.h   CInternetSession mySession;   CHttpFile* myHttpFile...

经验教程

257

收藏

15
微博分享 QQ分享 QQ空间 手机页面 收藏网站 回到头部