news 2026/1/19 9:09:04

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/1/16 10:16:45

Kotaemon餐厅推荐系统:结合口味偏好的个性化建议

Kotaemon餐厅推荐系统&#xff1a;结合口味偏好的个性化建议 在城市街头巷尾的餐饮选择越来越丰富的今天&#xff0c;用户早已不再满足于“附近有什么店开门”这种粗粒度的信息。他们真正关心的是&#xff1a;“有没有一家辣得够劲、价格合适、环境不吵、还能带朋友聚餐的川菜馆…

作者头像 李华
网站建设 2026/1/18 12:12:54

SpringAI搭建智能体(二):搭建客服系统智能体

本文详细介绍了智能体(Agent)的概念、特征及基于Spring AI的实现方案。智能体通过语义理解、任务拆解和工具调用等核心功能&#xff0c;能够自主执行复杂任务。文章展示了完整的智能体架构设计&#xff0c;包括工具接口、工具管理器、语义分析模块和智能体核心&#xff0c;并通…

作者头像 李华
网站建设 2026/1/17 3:04:06

革命性手势控制软件:重新定义Windows交互体验

革命性手势控制软件&#xff1a;重新定义Windows交互体验 【免费下载链接】GestureSign A gesture recognition software for Windows tablet 项目地址: https://gitcode.com/gh_mirrors/ge/GestureSign 在现代数字工作环境中&#xff0c;效率提升已成为技术爱好者的核心…

作者头像 李华
网站建设 2026/1/17 11:25:58

QtScrcpy终极优化指南:如何实现流畅高帧率手机投屏

你是否遇到过这样的困扰&#xff1a;在手机上玩游戏时画面丝滑流畅&#xff0c;但通过QtScrcpy投屏到电脑后却卡顿明显&#xff1f;特别是使用WASD移动键位映射时&#xff0c;操作会突然失效&#xff1f;别担心&#xff0c;这篇文章将为你提供完整的解决方案&#xff01; 【免费…

作者头像 李华
网站建设 2026/1/17 16:39:27

UnityPsdImporter实战攻略:告别手切PSD的烦恼时光

UnityPsdImporter实战攻略&#xff1a;告别手切PSD的烦恼时光 【免费下载链接】UnityPsdImporter Advanced PSD importer for Unity3D 项目地址: https://gitcode.com/gh_mirrors/un/UnityPsdImporter 还在为设计师发来的PSD文件而头疼吗&#xff1f;图层一个个导出、位…

作者头像 李华
网站建设 2026/1/17 6:42:34

【java入门到放弃】算法中队列

1、普通队列 FIFO、全称&#xff1a;First In, First Out、中文意思&#xff1a;先进先出 offer / poll / peek&#xff08;安全&#xff09;或 add / remove / element&#xff08;严格(为null会抛异常)&#xff09; public static void main(String[] args) {// 用 LinkedLis…

作者头像 李华