从汇编看c++中参数对象和局部对象的析构次序
添加时间:2013-5-14 点击量:
下面是c++的源码:
class X {
public:
int i;
int j;
~X() {}
};
void f(X x) {
X x1;
x.i = 1;
x.j = 2;
}
int main() {
f(X());
}
下面是main函数的汇编码:
_main PROC
; 15 : int main() {
push ebp
mov ebp, esp
sub esp, 8;为姑且对象预留8byte空间,因为没有显示定义机关函数,
;并且这种景象下编译器供给无用的默认机关函数,是以看不到机关函数的调用
; 16 : f(X());
mov eax, DWORD PTR ¥T2560[ebp+4];将偏移姑且变量的首地址4byte处内存中内容给eax,即将姑且变量的成员变量j值给eax
push eax;将eax压栈
mov ecx, DWORD PTR ¥T2560[ebp];将姑且变量首地址中的内容给ecx,即将姑且变量中的成员变量i值给ecx
push ecx;将ecx压栈
;上方四句创建了姑且变量的一份拷贝,作为参数调用f
call ?f@@YAXVX@@@Z ; 调用函数f
add esp, 8;将栈顶指针下移8byte,开释为参数对象的供给的栈空间
lea ecx, DWORD PTR ¥T2560[ebp];将姑且对象的首地址给ecx
call ??1X@@QAE@XZ ; 为姑且对象调用析构函数
; 17 : }
xor eax, eax
mov esp, ebp
pop ebp
ret 0
_main ENDP
从上方可以看出,产生的姑且对象在函数调用完成退出后才调用析构函数。
下面是f函数的汇编码:
?f@@YAXVX@@@Z PROC ; f
; 9 : void f(X x) {
push ebp
mov ebp, esp
sub esp, 8;为局部对象x1预留8byte的空间
; 10 : X x1;
; 11 : x.i = 1;
mov DWORD PTR _x¥[ebp], 1;把1写给参数对象首地址处,即把1写入参数对象的成员变量i
; 12 : x.j = 2;
mov DWORD PTR _x¥[ebp+4], 2;把2写入偏移参数对象首地址4byte处的内存,即把2写入参数对象的成员变量j
; 13 :
; 14 : }
lea ecx, DWORD PTR _x1¥[ebp];将局部变量x1的首地址给ecx
call ??1X@@QAE@XZ ; 为x1调用析构函数
lea ecx, DWORD PTR _x¥[ebp];将参数对象的首地址给ecx
call ??1X@@QAE@XZ ; 为参数对象调用析构函数
mov esp, ebp
pop ebp
ret 0
?f@@YAXVX@@@Z ENDP ; f
; Function compile flags: /Odtp
_TEXT ENDS
; COMDAT ??1X@@QAE@XZ
_TEXT SEGMENT
_this¥ = -4 ; size = 4
??1X@@QAE@XZ PROC ; X::~X, COMDAT
; _this¥ = ecx
; 6 : ~X() {}
push ebp
mov ebp, esp
push ecx
mov DWORD PTR _this¥[ebp], ecx
mov esp, ebp
pop ebp
ret 0
??1X@@QAE@XZ ENDP
从上方的代码可以看出,参数对象和局部对象都是在函数退出之前调用析构函数。并且参数对象在局部对象调用析构函数之后再调用本身的析构函数。
容易发怒的意思就是: 别人做了蠢事, 然后我们代替他们, 表现出笨蛋的样子。—— 蔡康永
下面是c++的源码:
class X {
public:
int i;
int j;
~X() {}
};
void f(X x) {
X x1;
x.i = 1;
x.j = 2;
}
int main() {
f(X());
}
下面是main函数的汇编码:
_main PROC
; 15 : int main() {
push ebp
mov ebp, esp
sub esp, 8;为姑且对象预留8byte空间,因为没有显示定义机关函数,
;并且这种景象下编译器供给无用的默认机关函数,是以看不到机关函数的调用
; 16 : f(X());
mov eax, DWORD PTR ¥T2560[ebp+4];将偏移姑且变量的首地址4byte处内存中内容给eax,即将姑且变量的成员变量j值给eax
push eax;将eax压栈
mov ecx, DWORD PTR ¥T2560[ebp];将姑且变量首地址中的内容给ecx,即将姑且变量中的成员变量i值给ecx
push ecx;将ecx压栈
;上方四句创建了姑且变量的一份拷贝,作为参数调用f
call ?f@@YAXVX@@@Z ; 调用函数f
add esp, 8;将栈顶指针下移8byte,开释为参数对象的供给的栈空间
lea ecx, DWORD PTR ¥T2560[ebp];将姑且对象的首地址给ecx
call ??1X@@QAE@XZ ; 为姑且对象调用析构函数
; 17 : }
xor eax, eax
mov esp, ebp
pop ebp
ret 0
_main ENDP
从上方可以看出,产生的姑且对象在函数调用完成退出后才调用析构函数。
下面是f函数的汇编码:
?f@@YAXVX@@@Z PROC ; f
; 9 : void f(X x) {
push ebp
mov ebp, esp
sub esp, 8;为局部对象x1预留8byte的空间
; 10 : X x1;
; 11 : x.i = 1;
mov DWORD PTR _x¥[ebp], 1;把1写给参数对象首地址处,即把1写入参数对象的成员变量i
; 12 : x.j = 2;
mov DWORD PTR _x¥[ebp+4], 2;把2写入偏移参数对象首地址4byte处的内存,即把2写入参数对象的成员变量j
; 13 :
; 14 : }
lea ecx, DWORD PTR _x1¥[ebp];将局部变量x1的首地址给ecx
call ??1X@@QAE@XZ ; 为x1调用析构函数
lea ecx, DWORD PTR _x¥[ebp];将参数对象的首地址给ecx
call ??1X@@QAE@XZ ; 为参数对象调用析构函数
mov esp, ebp
pop ebp
ret 0
?f@@YAXVX@@@Z ENDP ; f
; Function compile flags: /Odtp
_TEXT ENDS
; COMDAT ??1X@@QAE@XZ
_TEXT SEGMENT
_this¥ = -4 ; size = 4
??1X@@QAE@XZ PROC ; X::~X, COMDAT
; _this¥ = ecx
; 6 : ~X() {}
push ebp
mov ebp, esp
push ecx
mov DWORD PTR _this¥[ebp], ecx
mov esp, ebp
pop ebp
ret 0
??1X@@QAE@XZ ENDP
从上方的代码可以看出,参数对象和局部对象都是在函数退出之前调用析构函数。并且参数对象在局部对象调用析构函数之后再调用本身的析构函数。
容易发怒的意思就是: 别人做了蠢事, 然后我们代替他们, 表现出笨蛋的样子。—— 蔡康永