// Test1 typedef struct { int nValue; BYTE nValue2[4]; }ST_TEST;
int main() { ST_TEST sTest1 = {0}; ST_TEST sTest2 = {0}; sTest2 = sTest1; memcpy(&sTest2, &sTest1, sizeof(sTest1)); return 0; }
// Disassembly typedef struct { int nValue; BYTE nValue2[4]; }ST_TEST;
int main() { 004196B0 push ebp 004196B1 mov ebp,esp 004196B3 sub esp,0E4h 004196B9 push ebx 004196BA push esi 004196BB push edi 004196BC lea edi,[ebp-0E4h] 004196C2 mov ecx,39h 004196C7 mov eax,0CCCCCCCCh 004196CC rep stos dword ptr es:[edi] 004196CE mov eax,dword ptr [___security_cookie (428080h)] 004196D3 xor eax,ebp 004196D5 mov dword ptr [ebp-4],eax ST_TEST sTest1 = {0}; 004196D8 mov dword ptr [ebp-10h],0 004196DF xor eax,eax 004196E1 mov dword ptr [ebp-0Ch],eax ST_TEST sTest2 = {0}; 004196E4 mov dword ptr [ebp-20h],0 004196EB xor eax,eax 004196ED mov dword ptr [ebp-1Ch],eax sTest2 = sTest1; 004196F0 mov eax,dword ptr [ebp-10h] 004196F3 mov dword ptr [ebp-20h],eax 004196F6 mov ecx,dword ptr [ebp-0Ch] 004196F9 mov dword ptr [ebp-1Ch],ecx memcpy(&sTest2, &sTest1, sizeof(sTest1)); 004196FC push 8 004196FE lea eax,[ebp-10h] 00419701 push eax 00419702 lea ecx,[ebp-20h] 00419705 push ecx 00419706 call @ILT+980(_memcpy) (4113D9h) 0041970B add esp,0Ch
return 0; 0041970E xor eax,eax }
// Test2 typedef struct { int nValue; BYTE bValue[4092]; }ST_TEST;
int main() { ST_TEST sTest1 = {0}; ST_TEST sTest2 = {0}; sTest2 = sTest1; memcpy(&sTest2, &sTest1, sizeof(sTest1));
return 0; }
// Disassembly typedef struct { int nValue; BYTE bValue[4092]; }ST_TEST;
int main() { 004196B0 push ebp 004196B1 mov ebp,esp 004196B3 mov eax,20D4h 004196B8 call @ILT+415(__chkstk) (4111A4h) 004196BD push ebx 004196BE push esi 004196BF push edi 004196C0 lea edi,[ebp-20D4h] 004196C6 mov ecx,835h 004196CB mov eax,0CCCCCCCCh 004196D0 rep stos dword ptr es:[edi] 004196D2 mov eax,dword ptr [___security_cookie (428080h)] 004196D7 xor eax,ebp 004196D9 mov dword ptr [ebp-4],eax ST_TEST sTest1 = {0}; 004196DC mov dword ptr [ebp-1008h],0 004196E6 push 0FFCh 004196EB push 0 004196ED lea eax,[ebp-1004h] 004196F3 push eax 004196F4 call @ILT+510(_memset) (411203h) 004196F9 add esp,0Ch ST_TEST sTest2 = {0}; 004196FC mov dword ptr [ebp-2010h],0 00419706 push 0FFCh 0041970B push 0 0041970D lea eax,[ebp-200Ch] 00419713 push eax 00419714 call @ILT+510(_memset) (411203h) 00419719 add esp,0Ch sTest2 = sTest1; 0041971C push 1000h 00419721 lea eax,[ebp-1008h] 00419727 push eax 00419728 lea ecx,[ebp-2010h] 0041972E push ecx 0041972F call @ILT+980(_memcpy) (4113D9h) 00419734 add esp,0Ch memcpy(&sTest2, &sTest1, sizeof(sTest1)); 00419737 push 1000h 0041973C lea eax,[ebp-1008h] 00419742 push eax 00419743 lea ecx,[ebp-2010h] 00419749 push ecx 0041974A call @ILT+980(_memcpy) (4113D9h) 0041974F add esp,0Ch
return 0; 00419752 xor eax,eax }
通过上面的的两份测试代码,我们可以看出,结构的直接赋值(默认结构体赋值函数)与memcpy之间的异同. 例1中,结构体小于4096时,赋值是通过结构体中每个变量的赋值来完成的. 例2中,结构体大于等于4096时,结构体赋值与memcpy的效果是一样的.因为我们通过代码的反汇编,可以看出编译器已经将代码优化成一样的汇编代码了. 所以,理论上来说,针对结构体调用memcpy的效果与直接赋值基本上是一样的. 而且按照汇编的结果,直接赋值的汇编代码少于memcpy的汇编代码,所以更推荐使用结构体的直接赋值(注意,这里的结构体是指没有重载赋值函数的) |