Я попытался сравнить производительность встроенного языка ассемблера и кода на С++, поэтому я написал функцию, которая добавляет два массива размером 2000 в 100000 раз. Здесь код:
#define TIMES 100000
void calcuC(int *x,int *y,int length)
{
for(int i = 0; i < TIMES; i++)
{
for(int j = 0; j < length; j++)
x[j] += y[j];
}
}
void calcuAsm(int *x,int *y,int lengthOfArray)
{
__asm
{
mov edi,TIMES
start:
mov esi,0
mov ecx,lengthOfArray
label:
mov edx,x
push edx
mov eax,DWORD PTR [edx + esi*4]
mov edx,y
mov ebx,DWORD PTR [edx + esi*4]
add eax,ebx
pop edx
mov [edx + esi*4],eax
inc esi
loop label
dec edi
cmp edi,0
jnz start
};
}
Здесь main()
:
int main() {
bool errorOccured = false;
setbuf(stdout,NULL);
int *xC,*xAsm,*yC,*yAsm;
xC = new int[2000];
xAsm = new int[2000];
yC = new int[2000];
yAsm = new int[2000];
for(int i = 0; i < 2000; i++)
{
xC[i] = 0;
xAsm[i] = 0;
yC[i] = i;
yAsm[i] = i;
}
time_t start = clock();
calcuC(xC,yC,2000);
// calcuAsm(xAsm,yAsm,2000);
// for(int i = 0; i < 2000; i++)
// {
// if(xC[i] != xAsm[i])
// {
// cout<<"xC["<<i<<"]="<<xC[i]<<" "<<"xAsm["<<i<<"]="<<xAsm[i]<<endl;
// errorOccured = true;
// break;
// }
// }
// if(errorOccured)
// cout<<"Error occurs!"<<endl;
// else
// cout<<"Works fine!"<<endl;
time_t end = clock();
// cout<<"time = "<<(float)(end - start) / CLOCKS_PER_SEC<<"\n";
cout<<"time = "<<end - start<<endl;
return 0;
}
Затем я запускаю программу пять раз, чтобы получить циклы процессора, которые можно было увидеть как время. Каждый раз, когда я вызываю только одну функцию, указанную выше.
И вот результат.
Функция версии сборки:
Debug Release
---------------
732 668
733 680
659 672
667 675
684 694
Average: 677
Функция версии С++:
Debug Release
-----------------
1068 168
999 166
1072 231
1002 166
1114 183
Average: 182
Код С++ в режиме выпуска почти в 3,7 раза быстрее, чем код сборки. Почему?
Я предполагаю, что код сборки, который я написал, не так эффективен, как код, созданный GCC. Трудно для обычного программиста, как я, писать код быстрее, чем его противник, сгенерированный компилятором. Это означает, что я не должен доверять производительности языка ассемблера, написанного моими руками, сосредоточиться на С++ и забыть о языке ассемблера?