如何獲得程序或者一段代碼運(yùn)行的時(shí)間?你可能說有專門的程序測試工具,確實(shí),不過你也可以在程序中嵌入?yún)R編代碼來實(shí)現(xiàn)。
在Pentium的指令系統(tǒng)中有一條指令可以獲得CPU內(nèi)部64位計(jì)數(shù)器的值,我們可以通過代碼兩次獲取該計(jì)數(shù)器的值而獲得程序或代碼運(yùn)行的時(shí)鐘周期數(shù),進(jìn)而通過你的cpu的頻率算出一個(gè)時(shí)鐘周期的時(shí)間,從而算出程序運(yùn)行的確切時(shí)間。
我們通過指令TDSIC來獲得cpu內(nèi)部計(jì)數(shù)器的值,指令TDSIC返回值放在EDX:EAX中,其中EDX中存放64位寄存器中高32位的值,EAX存放第32位的值.
下面看看實(shí)現(xiàn)的代碼:
//用匯編實(shí)現(xiàn)獲取一段代碼運(yùn)行的時(shí)間
#include<iostream>
using namespace std;
void GetClockNumber (long high, long low);
void GetRunTime();
int main()
{
long HighStart,LowStart,HighEnd,LowEnd;
long numhigh,numlow;
//獲取代碼運(yùn)行開始時(shí)cpu內(nèi)部計(jì)數(shù)器的值
__asm
{
RDTSC
mov HighStart, edx
mov LowStart, eax
}
for(int i= 0; i<100000; i++ )
{
for(int i= 0; i<100000; i++ )
{
}
}
//獲取代碼結(jié)束時(shí)cpu內(nèi)部計(jì)數(shù)器的值,并減去初值
__asm
{
RDTSC
mov HighEnd, edx
Mov LowEnd, eax
;獲取兩次計(jì)數(shù)器值得差
sub eax, LowStart
cmp eax, 0 ; 如果低32的差為負(fù)則求返,因?yàn)榈诙稳〉糜肋h(yuǎn)比第一次的大
jg L1
neg eax
jmp L2
L1: mov numlow, eax
L2: sbb edx, HighStart
mov numhigh, edx
}
//把兩個(gè)計(jì)數(shù)器值之差放在一個(gè)64位的整形變量中
//先把高32位左移32位放在64的整形變量中,然后再加上低32位
__int64 timer =(numhigh<<32) + numlow;
//輸出代碼段運(yùn)行的時(shí)鐘周期數(shù)
//以頻率1.1Gcpu為例,如果換計(jì)算機(jī)把其中的1.1改乘其它即可,因?yàn)橄嘈糯蠹业腸pu都應(yīng)該在1G以上 ^_^
cout<< (double) (timer /1.1/1000000000) << endl;
return 0;
}
這樣通過一條簡單的匯編指令就可以獲得程序或一段代碼的大概時(shí)間,不過并不能得到運(yùn)行的確切時(shí)間,因?yàn)榧词谷サ糁虚g的循環(huán),程序也會(huì)有個(gè)運(yùn)行時(shí)間,
因?yàn)樵诘谝淮稳〉糜?jì)數(shù)器的值后,有兩條匯編指令mov HighStart, edx mov LowStart, eax這兩條指令當(dāng)然也有運(yùn)行時(shí)間 ,當(dāng)然你可以減去這兩條指令的運(yùn)行時(shí)間(在1.1G的機(jī)子上是3e-8s),這樣會(huì)更精確一點(diǎn)。
如果你要確切知道程序的運(yùn)行時(shí)間,專業(yè)的測試軟件肯定會(huì)更好一點(diǎn),不過好像一般沒有必要獲取除非專門的要求的程序。
不過能DIY一個(gè)也是不錯(cuò)的,不管有沒有,最起碼你可以學(xué)到在VC++中如何嵌入?yún)R編代碼以及如何使用32位的寄存器,其實(shí)和16位的寄存器一樣使用,將來64的也應(yīng)該一樣,只不過位數(shù)不同罷了。