news 2026/4/22 13:10:57

C++ 拷贝构造函数到底什么时候被调用?看这 3 种典型场景(附完整示例)

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
C++ 拷贝构造函数到底什么时候被调用?看这 3 种典型场景(附完整示例)

🧠 C++ 拷贝构造函数到底什么时候被调用?看这 3 种典型场景(附完整示例)

在 C++ 中,拷贝构造函数(Copy Constructor)是对象复制时的关键机制。很多初学者容易混淆“初始化”和“赋值”,也不清楚函数传参或返回时是否真的触发了拷贝。今天我们结合一段经典代码,彻底讲清楚!

🔧 示例类定义
#include <iostream> usingnamespacestd; class Person { public: Person() { cout << "无参构造函数!" << endl; mAge = 0; } Person(int age) { cout << "有参构造函数!" << endl; mAge = age; } Person(const Person& p) { cout << "拷贝构造函数!" << endl; mAge = p.mAge; } ~Person() { cout << "析构函数!" << endl; } public: int mAge; };

这个类能清晰打印出每种构造/析构的调用过程,非常适合教学。


✅ 场景一:用已有对象初始化新对象

void test01() { Person p1(20); // 有参构造 Person p2(p1); // 👉 调用拷贝构造 //Person newman2 = man; // 👉 也调用拷贝构造(等价于上一行) // ❌ 注意:以下不是拷贝构造! // Person newman3; // newman3 = man; // 这是赋值操作,调用 operator= }

✨ 关键点:只有在对象“创建时”用另一个对象初始化,才触发拷贝构造
A = B如果 A 已存在,就是赋值,不是构造!


✅ 场景二:函数参数按值传递

void doWork(Person p1) {} // 参数是值传递 void test02() { Person p; // 无参构造 doWork(p); // 👉 调用拷贝构造,为 p1 创建副本 }

💡 如果你看到函数内部修改了p1但不影响原对象,就是因为这里拷贝了一份。
想避免拷贝?改用const Person& p1


✅ 场景三:函数按值返回局部对象

Person doWork2() { Person p1; cout << "局部对象地址: " << &p1 << endl; return p1; // 理论上应拷贝 } void test03() { Person p = doWork2(); // 👉 理论上调用拷贝构造 cout << "外部对象地址: " << &p << endl; }

⚠️但实际运行时,你可能看不到“拷贝构造函数!”的输出!
原因:现代编译器会进行返回值优化(RVO),C++17 更是强制省略拷贝(guaranteed copy elision)。
所以p直接在doWork2()中构造,零拷贝

🔬 想验证拷贝是否发生?编译时加-fno-elide-constructors(GCC/Clang)即可关闭优化。


📌 总结:拷贝构造的三大调用时机

场景

是否调用拷贝构造

说明

Person p2(p1)

Person p2 = p1

✅ 是

对象初始化

函数参数按值传递foo(p1)

✅ 是

创建形参副本

函数返回局部对象return obj

❓ 可能被优化

C++17 起通常不调用

p2 = p1;(已存在对象)→ 调用赋值运算符,不是拷贝构造


💡 小贴士

  • 如果你的类管理资源(如指针、文件句柄),必须自定义拷贝构造,否则浅拷贝会导致 double-free 等严重 bug。

  • C++11 后还可定义移动构造函数,进一步提升性能。

  • 编译器优化是好事,但理解底层语义才能写出安全高效的代码!


通过这段代码 + 三个测试函数,你就能彻底掌握拷贝构造的调用逻辑。快去试试test01()test02()test03(),观察输出吧!

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/18 8:40:16

年会抽奖还在PPT?使用极空间NAS打造科技感爆棚的3D动态抽奖项目

年会抽奖还在PPT&#xff1f;使用极空间NAS打造科技感爆棚的3D动态抽奖项目 哈喽小伙伴们好&#xff0c;我是Stark-C~ 时间过得真快&#xff01;这不&#xff0c;又到年底了&#xff01;而在这岁末的热闹氛围中&#xff0c;最让人期待的&#xff0c;莫过于公司一年一度的年会…

作者头像 李华
网站建设 2026/4/18 8:18:07

WonderTrader终极指南:一站式量化交易策略开发框架

在金融科技快速发展的今天&#xff0c;量化交易已成为投资领域的重要趋势。WonderTrader作为一款开源、跨平台的金融交易策略开发框架&#xff0c;为您提供了从策略研发到实盘交易的全链路解决方案。无论您是量化交易初学者还是专业开发者&#xff0c;都能在这个平台上找到适合…

作者头像 李华
网站建设 2026/4/20 14:17:27

SM3-PHP终极指南:PHP开发者的国密加密实战手册

SM3-PHP终极指南&#xff1a;PHP开发者的国密加密实战手册 【免费下载链接】SM3-PHP 国密标准SM3的PHP实现 项目地址: https://gitcode.com/gh_mirrors/sm3/SM3-PHP 你是否曾在PHP项目中遇到这样的困境&#xff1a;需要实现国密标准的加密算法&#xff0c;却苦于找不到合…

作者头像 李华
网站建设 2026/4/18 4:32:08

Great Expectations数据验证终极指南:从基础到实战应用

Great Expectations数据验证终极指南&#xff1a;从基础到实战应用 【免费下载链接】great_expectations Always know what to expect from your data. 项目地址: https://gitcode.com/GitHub_Trending/gr/great_expectations 在数据驱动的业务环境中&#xff0c;数据质…

作者头像 李华
网站建设 2026/4/18 4:29:48

如何快速配置Nexe:Node.js应用打包的完整指南

如何快速配置Nexe&#xff1a;Node.js应用打包的完整指南 【免费下载链接】nexe &#x1f389; create a single executable out of your node.js apps 项目地址: https://gitcode.com/gh_mirrors/ne/nexe Nexe是一个强大的Node.js应用打包工具&#xff0c;能够将你的整…

作者头像 李华