当前位置: 首页 > news >正文

深入浅出:C++深拷贝与浅拷贝

🚀 C++ 深拷贝与浅拷贝详解:从原理到实战
📅 更新时间:2025年6月11日
🏷️ 标签:C++ | 深拷贝 | 浅拷贝 | 对象拷贝 | 内存管理 | C++基础

文章目录

  • 前言
  • 📌 一、拷贝基础:浅拷贝与深拷贝究竟是什么?
    • 1.浅拷贝(Shallow Copy)
      • 示例:
    • 2.深拷贝(Deep Copy)
      • 示例:
  • 🔧二、为什么需要关注深拷贝与浅拷贝?
  • 📂三、 案例详解浅拷贝与深拷贝
    • 1.案例背景
    • 2.问题示例
      • 解决方案一
      • 解决方案二
      • 解决方案三 (解决本质问题)
  • 📝总结


前言

在 C++ 编程中,对象的复制操作看似简单,但其中隐藏的内存管理问题往往是初学者最容易踩的地雷。这篇文章将以循序渐进的方式,从最基础的值拷贝地址拷贝讲起,详细解读浅拷贝深拷贝的原理、区别以及应用场景,帮助你构建正确的拷贝语义认识,为后续的封装面向对象知识打下坚实基础


📌 一、拷贝基础:浅拷贝与深拷贝究竟是什么?

1.浅拷贝(Shallow Copy)

浅拷贝是指在复制对象时,只复制对象的基本数据成员(值拷贝),而对于指针成员,仅复制指针的值(即内存地址),而不复制指针所指向的内存内容。换句话说,源对象和新对象共享同一块动态分配的内存

示例:

#include<iostream>
using namespace std;class house
{public:house(int sum):area(sum){p=&area;}~house(){}void where(){cout<<"I live in "<<p<<endl;}int area;int* p;
};int main()
{house a(100);a.where();house b=a;b.where();return 0;
}

输出:

I live in 0x7ffe702f1be0
I live in 0x7ffe702f1be0

我们会发现,house 对象a对象b共享同一块地址这就是浅拷贝

2.深拷贝(Deep Copy)

深拷贝是指在复制对象时,不仅复制对象的基本数据成员,还为指针成员分配新的内存,并将源对象指针指向的内容复制到新分配的内存中。新对象和源对象完全独立,互不影响

示例:

输入:

#include<iostream>
using namespace std;int main()
{int a=10;int b=a;cout<<"a地址:"<<&a<<endl;cout<<"b地址:"<<&b<<endl;return 0;
}

输出:

a地址:0x7fffb62b7a80
b地址:0x7fffb62b7a84

我们会发现a和b是独立分开的,这就是深拷贝

🔧二、为什么需要关注深拷贝与浅拷贝?

在 C++ 中,类的默认拷贝行为(由编译器提供的默认拷贝构造函数和赋值运算符)是浅拷贝。如果你的类管理动态内存(如指针成员),默认的浅拷贝可能导致以下问题:

1.悬空指针(Dangling Pointer):

如果两个对象共享同一块内存,当一个对象销毁时释放了这块内存,另一个对象的指针会变成悬空指针,访问它会导致未定义行为。

2.内存泄漏(Memory Leak):

如果没有正确管理动态内存,可能会导致内存无法释放。

3.数据不一致:

修改一个对象的动态内存内容会意外影响另一个对象。

因此,在涉及动态内存的类中,通常需要显式实现深拷贝,以确保对象的独立性和程序的正确性

📂三、 案例详解浅拷贝与深拷贝

1.案例背景

此时有一个类,里面包含两个变量,一个是面积area,一个是指向area地址的指针p
相当于此时你需要去购买房子,房子的面积为area,此房子所配的钥匙为指针p

假设我们的类是这样写的!假设!

class house
{public:house(int sum):area(sum){p=malloc(area);//给所买的房子分配一把钥匙}~house(){free(p);p=nullptr;}void where(){cout<<"I live in "<<p<<endl;}int area;//房屋面积void* p;//指向房子的地址  相当于钥匙
};

2.问题示例

如果对象a购买了一处房产,b也想购买一处与a一样大小的房产,下面写法会不会出错?为什么?

#include<iostream>
using namespace std;class house
{public:house(int sum):area(sum){p=malloc(area);}~house(){free(p);p=nullptr;}void where(){cout<<"I live in "<<p<<endl;}int area;void* p;
};int main()
{house a(100);a.where();house b=a;//本意是想购买与a房子同样大小的房子b.where();return 0;
}

输出:

I live in 0x55ce6707deb0
I live in 0x55ce6707deb0
free(): double free detected in tcache 2!!!!!报错

为什么会错呢?我们仔细看看对象a和对象b的地址,发现都相同,说明此时共享同一块内存地址,当对象a进行析构的时候会将指针置空,但轮到对象b析构的时候,就产生了对空指针再次清除的报错

解决方案一

我们在构造对象b的时候换一种方式就行

	house a(100);a.where();house b(100);b.where();

输出:

I live in 0x55efdd689eb0
I live in 0x55efdd68a330

解决方案二

我们修改类,不用mallco去主动为p分配内存

#include<iostream>
using namespace std;class house
{public:house(int sum):area(sum){p=&area;}~house(){}void where(){cout<<"I live in "<<p<<endl;}int area;int* p;
};int main()
{house a(100);a.where();house b=a;b.where();return 0;
}输出:
I live in 0x7ffd4341c5a0
I live in 0x7ffd4341c5a0

但这样并没有解决问题的本质,本质上对象a和对象b还是同一个内存

解决方案三 (解决本质问题)

本质问题是一个深拷贝和浅拷贝的问题,我们要进行深拷贝,要独立地址
所以我们在类里面可以这样修改

#include<iostream>
using namespace std;class house
{public:house(int sum):area(sum){p=&area;}house(const house&a):area(a.area)//深拷贝构造{p=&area;}~house(){}void where(){cout<<"I live in "<<p<<endl;}int area;int* p;
};int main()
{house a(100);a.where();house b=a;b.where();return 0;
}输出:
I live in 0x7ffdf435f970
I live in 0x7ffdf435f980

这样就解决这个问题了

通过简单的案例来理解深拷贝与浅拷贝,本质上就是是否共享同一个内存区域的问题

📝总结

浅拷贝 适用于仅含基本数据类型的对象;

深拷贝 适用于含有动态内存或指针的对象,防止资源重复释放或共享;

如果你觉得本文对你有帮助,不妨点赞 + 收藏 + 关注,更多 C++ 系列教程将持续更新 🔥!

相关文章:

  • Web防火墙深度实战:从漏洞修补到CC攻击防御
  • 重拾前端基础知识:CSS预处理器
  • 基于AI智能体的医疗AI工具库构建路径分析
  • Python爬虫(54)Python数据治理全攻略:从爬虫清洗到NLP情感分析的实战演进
  • 第七章: SEO与渲染方式 三
  • C#接口代码记录
  • 第七章: SEO与渲染方式
  • Scrapy爬虫框架:数据采集的瑞士军刀(附实战避坑指南)!!!
  • ( github actions + workflow 01 ) 实现爬虫自动化,每2小时爬取一次澎湃新闻
  • MyBatis实战指南(七)MyBatis缓存机制
  • Python毕业设计226—基于python+爬虫+html的豆瓣影视数据可视化系统(源代码+数据库+万字论文)
  • Linux:多线程---线程控制(线程创建线程等待线程终止)
  • AJAX、Axios 与 Fetch:现代前端数据请求技术对比
  • Linux Swap分区应该禁用吗?深入辨析其作用与性能优化
  • React 状态管理指南:Redux 原理与优化策略
  • React删除评论逻辑:1、客户端立即更新UI(乐观更新)2、后台调用删除评论API
  • Flutter setState() 状态管理详细使用指南
  • 一键实现全站多语言化:translate.js 极简集成指南,支持Vue 、React 框架。
  • 《HarmonyOSNext属性动画实战手册:让UI丝滑起舞的魔法指南》
  • 深度学习小项目合集之音频语音识别
  • 想接外包做网站/微信推广软件
  • 摄影网站源码 国外/电商运营主要工作内容
  • 做盗版网站会怎样/宁波seo教程网
  • 手表网站 二手/搜狗站长管理平台
  • 高端家具东莞网站建设技术支持/seo网站优化服务商
  • 郑州网站建设 郑州网站设计/百度权重排名