The Semantics of Constructors(拷贝构造函数之编译背后的行为)

本文探讨了C++中拷贝构造函数的三种应用场景及其默认实现方式。当类未显式提供拷贝构造函数时,编译器将采取默认成员初始化或递归初始化等方式进行对象复制。此外,还讨论了非位复制语义的情况及如何避免浅拷贝导致的未定义行为。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

本文是 Inside The C++ Object Model's Chapter 2  的部分读书笔记。

有三种情况,需要拷贝构造函数:

1)object直接为另外一个object的初始值

2)object作为函数以值传递的参数

3) object以函数返回值形式返回

 

如 果class没有提供一个explicit copy constructor时,编译器会以default memberwise initialization,也就是把每一个内建的或者派生的data member的值,从某个object直接拷贝到另外一个object上,对于member class object,以递归的方式施行memberwise initialization。

编译器并不会生成trivial的copy constructor,除非是bitwise copy semantics,以下四种情况就是非bitwise copy semantics:

1)class的某个member object声明有一个copy constructor时,这个copy constructor不会是被设计者声明的,或者是被编译器合成的

2)class继承自的base class含有copy constructor时

3)class含有virtual functions

4)class的base class是virtual时

编译器需要合成copy constructor,并且调用member object或者base class的copy constructor。

 

对 于virtual function或者virtual inheritance,编译器需要为每一个object设计好vptr以便在runtime获取到正确的virtual function table。因为vptr处理不好有可能发生对象的sliced,导致出现blow up。

 

总结:

如 果class是bitwise copy semantics,那么编译器不会合成copy constructor,只是按照default memberwise initialization copy source object到destination object。否则,编译器将合成copy constructor,以调用member object或者base class的copy constructor或者使得vptr设置好,使得在runtime可以使virtual base class或者base class可以正确访问virtual functions。

 

注意,本文只是说明编译器在copy constructor上为我们做了什么,并不是编译器合成的copy constructor就是安全可用的。这里需要和deep copy(深拷贝)/shallow copy(浅拷贝)加以区别。和编译器是否为我们合成default constructor一样,它只是完成自己的职责,并不会为non class member object and non-static members 初始化。 如果需要deep copy,设计者必须explicit declare copy constructor。否则即使编译器合成了,也会使得程序出现undefined behaviors。

转载于:https://www.cnblogs.com/anzhsoft/p/3464327.html

### C++ 拷贝构造函数与默认构造函数的区别 #### 定义差异 拷贝构造函数是一种特殊的构造函数,它通过已存在的同类对象来初始化新的对象实例。当一个对象作为参数传递给另一个函数或返回时,通常会触发拷贝构造函数[^1]。 ```cpp class MyClass { public: // 拷贝构造函数定义 MyClass(const MyClass& other); }; ``` 而默认构造函数则是在没有任何参数的情况下被调用来创建一个新的对象实例。其主要作用是提供一种方式,在不指定任何初始值的前提下构建对象。 ```cpp class MyClass { public: // 默认构造函数定义 MyClass(); }; ``` #### 行为区别 对于未显式声明拷贝构造函数的情况,编译器将会合成一个简单的成员逐个复制版本。这意味着所有非静态数据成员都将按照字面意义进行赋值操作: ```cpp // 编译器自动生成的拷贝构造函数行为类似于下面这样: MyClass::MyClass(const MyClass& other) : member(other.member), anotherMember(other.anotherMember) {} ``` 相比之下,默认构造函数可以为空实现,也可以设置某些合理的缺省状态;如果没有用户提供的默认构造函数,则编译器也会生成一个什么都不做的默认构造函数。 ```cpp // 用户可能定义如下形式的默认构造函数 MyClass::MyClass() : member(defaultValue), anotherMember(anotherDefaultValue) {} // 或者完全依赖于编译器隐含提供的无参构造函数 ``` #### 使用场景对比 - **深拷贝 vs 浅拷贝** 当涉及到指针或其他复杂类型的成员变量时,使用默认的浅拷贝可能会导致多个对象共享同一份实际的数据存储区域,这往往不是期望的结果。因此在这种情况下应当重载拷贝构造函数以执行所谓的“深拷贝”,即真正独立地复制底层资源[^2]。 - **性能考虑** 对于大型对象或者含有昂贵计算逻辑的对象来说,频繁调用拷贝构造函数可能导致不必要的开销。此时可以通过引入移动语义(move semantics),利用右值引用特性优化效率问题。 - **防止意外复制** 如果某个类设计不允许发生任何形式上的复制动作——比如独占型资源管理者——应该删除或私有化该类的拷贝构造函数并阻止外部访问。 ```cpp class NonCopyableResourceHolder { private: NonCopyableResourceHolder(const NonCopyableResourceHolder&) = delete; public: // ...其他成员... }; ```
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值