У меня есть этот код:
class Class {
public:
virtual void first() {};
virtual void second() {};
};
Class* object = new Class();
object->first();
object->second();
delete object;
что я компилирую с помощью Visual C++ 10 с /O2 и имею эту дизассемблирование:
282: Class* object = new Class();
00403953 push 4
00403955 call dword ptr [__imp_operator new (4050BCh)]
0040395B add esp,4
0040395E test eax,eax
00403960 je wmain+1Ch (40396Ch)
00403962 mov dword ptr [eax],offset Class::`vftable' (4056A4h)
00403968 mov esi,eax
0040396A jmp wmain+1Eh (40396Eh)
0040396C xor esi,esi
283: object->first();
0040396E mov eax,dword ptr [esi]
00403970 mov edx,dword ptr [eax]
00403972 mov ecx,esi
00403974 call edx
284: object->second();
00403976 mov eax,dword ptr [esi]
00403978 mov edx,dword ptr [eax+4]
0040397B mov ecx,esi
0040397D call edx
285: delete object;
0040397F push esi
00403980 call dword ptr [__imp_operator delete (405138h)]
Обратите внимание, что в 00403968
адрес начала объекта (где хранится vptr
) копируется в регистр esi
. Затем в 0040396E
этот адрес используется для получения vptr
, а значение vptr
используется для получения адреса first()
. Затем в 00403976
снова извлекается vptr
и используется для получения адреса second()
.
Почему vptr извлекается дважды? Возможно ли, что объект vptr
изменился между вызовами, или это просто недооптимизация?
esi
. - person Luchian Grigore   schedule 18.09.2012