构造函数在对象创建时调用,析构函数在对象生命周期结束时调用,两者在struct和class中行为一致,调用时机取决于对象的存储类型和作用域。

C++中,结构体(struct)的构造函数和析构函数何时被调用,核心逻辑其实与类(class)完全一致:构造函数在对象被创建时执行,而析构函数在对象生命周期结束时执行。这听起来很简单,但实际操作中,根据对象的存储类型和创建方式不同,具体的调用时机还是有不少细节值得琢磨的。
简单来说,构造函数在结构体对象被实例化时自动调用,无论这个对象是在栈上、堆上、作为全局变量、静态变量,还是作为另一个对象的成员。它的职责是确保对象在被使用前处于一个有效的、可预测的状态。而析构函数则在对象生命周期结束时被调用,负责清理对象占用的资源,比如释放动态分配的内存、关闭文件句柄等。
具体到不同的场景:
栈上对象(局部变量):
立即学习“C++免费学习笔记(深入)”;
break
continue
goto
堆上对象(动态分配):
new
delete
delete
全局对象和静态对象:
main
static
main
main
作为其他对象的成员:
临时对象:
说实话,很多人在刚接触C++的时候,都会纠结结构体(struct)和类(class)到底有什么本质区别。从构造函数和析构函数的调用机制来看,我可以很肯定地告诉你,它们没有任何区别。在C++标准中,
struct
class
struct
public
class
private
除此之外,它们在行为上完全一致。这意味着,你为
struct
class
struct
class
我个人认为,C++保留
struct
struct
class
struct
#include <iostream>
#include <string>
struct MyStruct {
std::string name;
MyStruct(const std::string& n) : name(n) {
std::cout << "MyStruct " << name << " Constructed." << std::endl;
}
~MyStruct() {
std::cout << "MyStruct " << name << " Destructed." << std::endl;
}
};
class MyClass {
public: // 必须显式声明public,否则默认是private
std::string name;
MyClass(const std::string& n) : name(n) {
std::cout << "MyClass " << name << " Constructed." << std::endl;
}
~MyClass() {
std::cout << "MyClass " << name << " Destructed." << std::endl;
}
};
void testFunction() {
MyStruct s_local("local_struct");
MyClass c_local("local_class");
} // s_local 和 c_local 在这里析构
int main() {
std::cout << "--- Entering main ---" << std::endl;
// 栈上对象
MyStruct s1("stack_struct_1");
MyClass c1("stack_class_1");
// 堆上对象
MyStruct* ps = new MyStruct("heap_struct");
MyClass* pc = new MyClass("heap_class");
testFunction(); // 调用函数,局部对象在这里构造和析构
delete ps; // 释放堆上对象
delete pc; // 释放堆上对象
std::cout << "--- Exiting main ---" << std::endl;
return 0;
} // s1 和 c1 在这里析构从上面的代码运行结果,你就能清楚地看到,
MyStruct
MyClass
虽然构造函数和析构函数的调用规则看起来很直接,但在一些特殊场景下,它们的行为确实可能与我们直觉上的预期有所偏差,这往往也是C++初学者容易踩坑的地方。
Placement New:这是一个比较高级的特性。
placement new
new (address) Type(args)
new
placement new
delete
placement new
object_ptr->~Type();
异常安全与构造失败:如果一个对象的构造函数在执行过程中抛出了异常,那么这个对象可能并没有完全构造成功。在这种情况下,C++运行时会确保已经成功构造的子对象(如果它有成员对象)的析构函数会被调用,以避免资源泄漏。但是,抛出异常的那个对象的析构函数本身不会被调用,因为它根本就没有成功完成构造。这对于编写异常安全的代码至关重要,要求我们在构造函数中分配的资源,要么在构造失败时能自动回滚,要么通过RAII(Resource Acquisition Is Initialization)机制来管理。
容器操作与拷贝/移动语义:当你使用
std::vector
std::list
push_back
emplace_back
push_back
emplace_back
std::vector
拷贝省略(Copy Elision)/返回值优化(RVO):现代C++编译器非常智能,它们可能会为了优化性能,省略掉某些不必要的拷贝构造函数和析构函数的调用。例如,当一个函数返回一个对象时,编译器可能会直接在调用者的栈帧上构造这个对象,而不是先在函数内部构造一个临时对象,再拷贝(或移动)出来。这被称为返回值优化(RVO)。虽然这通常是好事,因为它提高了效率,但如果你在构造函数或析构函数中依赖某些副作用来追踪对象的生命周期,可能会发现有些调用“消失”了。
联合体(Union):联合体允许在同一块内存中存储不同的数据成员,但一次只能有一个成员是活跃的。如果联合体包含非POD(Plain Old Data)类型,特别是带有自定义构造函数和析构函数的结构体,情况就会变得非常复杂。你不能直接为联合体定义析构函数来清理所有成员,因为你不知道哪个成员是活跃的。通常,你需要手动追踪哪个成员是活跃的,并在必要时手动调用其析构函数。这是C++中一个比较棘手且容易出错的特性。
这些场景都提醒我们,理解C++对象生命周期的底层机制,而不是仅仅依赖表面现象,是多么重要。
在我日常开发中,追踪构造函数和析构函数的调用顺序是排查对象生命周期问题、内存泄漏或者理解复杂系统行为的常用手段。这里有一些我个人觉得非常有效的方法:
利用 std::cout
std::cout
struct MyObject {
int id;
MyObject(int i) : id(i) {
std::cout << "Constructing MyObject " << id << " at " << this << std::endl;
}
~MyObject() {
std::cout << "Destructing MyObject " << id << " at " << this << std::endl;
}
};这种方式的缺点是会污染代码,但对于快速定位问题,它简直是神器。在大型项目中,我会用一个统一的日志宏来替代
std::cout
使用调试器(Debugger):这是专业开发者的必备工具。在构造函数和析构函数的第一行设置断点。当程序执行到这些断点时,你可以:
std::cout
利用RAII原则进行资源管理:RAII(Resource Acquisition Is Initialization,资源获取即初始化)是C++中一个非常核心的编程范式。它的基本思想是,将资源的生命周期绑定到对象的生命周期上。当对象被构造时,它获取资源;当对象被析构时,它释放资源。
std::unique_ptr
std::shared_ptr
new
delete
std::unique_ptr
内存泄漏检测工具:像Valgrind (Linux/macOS) 或 AddressSanitizer (ASan,GCC/Clang) 这样的工具,虽然主要用于检测内存错误,但它们也能间接帮助你追踪析构函数的调用问题。如果一个堆上分配的对象没有被
delete
自定义分配器或全局 new
delete
operator new
operator delete
operator new
delete
在我看来,没有银弹,通常是组合使用这些方法。对于快速验证,
std::cout
以上就是C++中结构体的构造函数和析构函数何时会被调用的详细内容,更多请关注php中文网其它相关文章!
每个人都需要一台速度更快、更稳定的 PC。随着时间的推移,垃圾文件、旧注册表数据和不必要的后台进程会占用资源并降低性能。幸运的是,许多工具可以让 Windows 保持平稳运行。
Copyright 2014-2025 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号