从汇编看c++中参数对象和局部对象的析构顺序
下面是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++
标签:
参数对象 局部对象 顺序
【推荐】国内首个AI IDE,深度理解中文开发场景,立即下载体验Trae
【推荐】编程新体验,更懂你的AI,立即体验豆包MarsCode编程助手
【推荐】抖音旗下AI助手豆包,你的智能百科全书,全免费不限次数
【推荐】轻量又高性能的 SSH 工具 IShell:AI 加持,快人一步
· 记一次.NET内存居高不下排查解决与启示
· 探究高空视频全景AR技术的实现原理
· 理解Rust引用及其生命周期标识(上)
· 浏览器原生「磁吸」效果!Anchor Positioning 锚点定位神器解析
· 没有源码,如何修改代码逻辑?
· 全程不用写代码,我用AI程序员写了一个飞机大战
· MongoDB 8.0这个新功能碉堡了,比商业数据库还牛
· 记一次.NET内存居高不下排查解决与启示
· DeepSeek 开源周回顾「GitHub 热点速览」
· 白话解读 Dapr 1.15:你的「微服务管家」又秀新绝活了