QQ登录

只需要一步,快速开始

 注册地址  找回密码
查看: 3808|回复: 2
打印 上一主题 下一主题

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

[复制链接]
字体大小: 正常 放大
xShandow        

43

主题

1

听众

385

积分

升级  28.33%

该用户从未签到

国际赛参赛者

新人进步奖

跳转到指定楼层
1#
发表于 2004-9-9 11:14 |只看该作者 |倒序浏览
|招呼Ta 关注Ta
如何获得程序或者一段代码运行的时间?你可能说有专门的程序测试工具,确实,不过你也可以在程序中嵌入汇编代码来实现。$ T  x( W! M3 p3 C
     在Pentium的指令系统中有一条指令可以获得CPU内部64位计数器的值,我们可以通过代码两次获取该计数器的值而获得程序或代码运行的时钟周期数,进而通
# {. ?8 S2 `/ H5 V<>过你的cpu的频率算出一个时钟周期的时间,从而算出程序运行的确切时间。# Q$ f, B$ h6 v
    我们通过指令TDSIC来获得cpu内部计数器的值,指令TDSIC返回值放在EDX:EAX中,其中EDX中存放64位寄存器中高32位的值,EAX存放第32位的值.</P>
- H( r8 f& w) a2 [$ p<>下面看看实现的代码:. F. j5 B6 c) S/ f
=============================================================================================" M% b) F7 u: q' j9 S. ~" T
//用汇编实现获取一段代码运行的时间</P>
( a+ X, X# B2 n; g$ T6 T<>#include&lt;iostream&gt;</P>0 `: v, s5 n  _/ B" l  E& L
<>using namespace std;</P>" U  B, ^8 Z$ H% r3 I
<>void GetClockNumber (long high, long low);
* p& W7 X8 N& `) Svoid GetRunTime();/ F; _: v, P5 M( E: p' D. ]
8 M7 u/ b5 f* c# I2 }) z
int main()& Q- G6 R4 K) N( M3 X
{      
" a$ F& s1 q2 w- m  d9 f7 D: j
6 k' V: q4 l$ Y/ O3 E2 a& N& f long HighStart,LowStart,HighEnd,LowEnd;9 Q. i9 e  N+ {
long numhigh,numlow;
* p6 W& [2 _& G1 o //获取代码运行开始时cpu内部计数器的值
# Q9 Q4 i" V. F% O __asm              2 ^' G& J6 s" D
{
/ s4 a2 L3 F& Z2 e3 I  RDTSC
! d& O1 T* t8 @+ O$ @  mov HighStart, edx: \1 s: k/ I' l0 x
  mov LowStart, eax
/ a7 d% v& j( O0 q, v  h8 q3 b }
" c) B8 T' }8 [2 `: P8 ^: g for(int i= 0; i&lt;100000; i++ )
" L  y8 A' X+ S0 F {
! o3 V/ f3 P! v# p2 s; L- M8 a             for(int i= 0; i&lt;100000; i++ )
: b7 a) M3 c1 F      {  }- x! P( q9 W* U% w+ Y' _% A" d
      
, Z5 ?* {; C- G/ b7 t- m      }
, M& O# ?5 Q* W2 c1 T( n }- p* a# r( f5 I% e. H, J2 q$ R" M
//获取代码结束时cpu内部计数器的值,并减去初值
' {- V0 h# t4 S% P# W; |/ c        __asm, t3 C9 g/ L, r# U
{
8 O  j4 E& M0 R* p2 l1 S# R  RDTSC$ z7 N# N+ e9 q; r
  mov HighEnd, edx
2 a" h" K) j' ]( @2 R  Mov LowEnd,  eax; {) x  v3 T( ~* C# ]
  ;获取两次计数器值得差
/ I. w* z# q/ V/ W6 _5 t  sub eax,  LowStart) l" _2 w( A4 b6 \- a( L3 X
  cmp    eax,  0       ; 如果低32的差为负则求返,因为第二次取得永远比第一次的大& U, F" n- C9 B4 J
  jg     L1
& S4 z/ e9 R6 S5 ]5 ]1 C) [% V  neg     eax # y# _% C  `" G  @
  jmp     L2: j5 J: ?  q5 f4 e3 k
            L1: mov numlow,  eax7 K5 b% H$ y9 H3 R' L
            L2: sbb edx,  HighStart
% Y- Q0 |( z% S, Q7 a8 f  mov numhigh, edx
7 y0 U$ b+ u% X1 z  4 ~2 ~/ T; u- w. x
}
# X& e  v8 e% G        //把两个计数器值之差放在一个64位的整形变量中
1 o0 \. O$ M; t3 L        //先把高32位左移32位放在64的整形变量中,然后再加上低32位% w8 l; b% [/ i2 l  [& d1 r2 z
__int64  timer =(numhigh&lt;&lt;32) + numlow;
% X* e5 s, I$ ?. v6 Q         //输出代码段运行的时钟周期数+ r( l" B) w+ J" g+ _/ L, G; z
         //以频率1.1Gcpu为例,如果换计算机把其中的1.1改乘其它即可,因为相信大家的cpu都应该在1G以上  ^_^ + s; G! Y/ n0 t) \- I) W7 f
cout&lt;&lt; (double) (timer /1.1/1000000000) &lt;&lt; endl;
% p6 |' D! ?! t3 O1 Z return 0;
' o( o$ z: ~8 v! a6 S; w}</P>
) ], }# d7 F+ C0 ~7 j4 K<>     这样通过一条简单的汇编指令就可以获得程序或一段代码的大概时间,不过并不能得到运行的确切时间,因为即使去掉中间的循环,程序也会有个运行时间,</P>5 r6 v' y8 g1 V( M& @2 ^  I/ [$ r
<>因为在第一次取得计数器的值后,有两条汇编指令mov HighStart, edx       mov LowStart, eax这两条指令当然也有运行时间 ,当然你可以减去这两条指令的运</P>
' I' n' }; J! O8 y' Z0 b! G<>行时间(在1.1G的机子上是3e-8s),这样会更精确一点。^_^% Q( S" O3 ^7 T# b1 i( h
    如果你要确切知道程序的运行时间,专业的测试软件肯定会更好一点,不过好像一般没有必要获取除非专门的要求的程序。不过能DIY一个也是不错的,不管有</P>
8 w+ S5 n/ R& M) S<>没有,最起码你可以学到在VC++中如何嵌入汇编代码以及如何使用32位的寄存器,其实和16位的寄存器一样使用,将来64的也应该一样,只不过位数不同罢了。
9 W( g7 x; n1 t; c- u0 ?( |& U^_^</P>
zan
转播转播0 分享淘帖0 分享分享0 收藏收藏0 支持支持0 反对反对0 微信微信
413009449        

0

主题

4

听众

43

积分

升级  40%

该用户从未签到

回复

使用道具 举报

黄窗帘        

0

主题

4

听众

28

积分

升级  24.21%

该用户从未签到

回复

使用道具 举报

您需要登录后才可以回帖 登录 | 注册地址

qq
收缩
  • 电话咨询

  • 04714969085
fastpost

关于我们| 联系我们| 诚征英才| 对外合作| 产品服务| QQ

手机版|Archiver| |繁體中文 手机客户端  

蒙公网安备 15010502000194号

Powered by Discuz! X2.5   © 2001-2013 数学建模网-数学中国 ( 蒙ICP备14002410号-3 蒙BBS备-0002号 )     论坛法律顾问:王兆丰

GMT+8, 2025-6-2 20:03 , Processed in 1.500206 second(s), 63 queries .

回顶部