C++中深拷贝与浅拷贝的区别

2024-04-11 23:04
文章标签 c++ 区别 拷贝 中深

本文主要是介绍C++中深拷贝与浅拷贝的区别,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

1.概念

深拷贝(deep copy)和浅拷贝(shallow copy)是在编程中经常遇到的概念,尤其在处理数据结构或对象时很重要。深拷贝和浅拷贝是相对于类的成员指针变量而言的,如果类内不存在成员指针变量,就没有深拷贝和浅拷贝。

  1. 浅拷贝:在浅拷贝中,只复制了对象本身,而不复制对象内部包含的引用对象。简而言之,新对象的引用仍然指向原始对象内部的引用对象。这意味着如果你修改了新对象内部的引用对象,原始对象也会受到影响,因为它们共享相同的引用。通常,浅拷贝使用简单的赋值操作来实现。

  2. 深拷贝:深拷贝会复制对象本身以及对象内部的所有引用对象,包括引用对象内部的引用对象,以此类推。这样,原始对象和深拷贝后的对象是完全独立的,对一个对象的修改不会影响另一个对象。深拷贝通常需要递归地复制所有引用对象,确保每个对象都是全新的。

深拷贝和浅拷贝在不同情况下有不同的应用场景。通常情况下:

  • 当你希望修改一个对象时,不影响原始对象,就应该使用深拷贝。
  • 当对象较大且内部结构较为简单时,浅拷贝可能更高效

2. 浅拷贝

浅拷贝 :拷贝对象的时候,不对指针成员所指向的对象进行拷贝,而是两个对象的指针成员指向同一个对象。
示例:
class Phone
{
private:string name;
public:Phone(string name):name(name){}
};class Person
{
private:Phone* pPhone;//当一个对象里有指针类型成员时,才有必要去考虑深拷贝和浅拷贝
public:Person(){pPhone = new Phone("诺基亚plus");}Phone* getPhone(){return pPhone;}
};int main()
{Person* xiaoming = new Person();Person xiaoqiang(*xiaoming);
//因Person中没有显式的定义拷贝构造函数,所以使用默认拷贝构造函数,默认拷贝构造函数的逻辑是浅拷贝cout<<xiaoming->getPhone()<<endl;//这两个对象的phone指针应该指向同一个对象cout<<xiaoqiang.getPhone()<<endl;
}

3.深拷贝

深拷贝:需要在对象的拷贝构造函数中对成员指针所指向的对象也进行拷贝。
#include <iostream>
using namespace std;
class Phone
{string name;
public:Phone(string name):name(name){}
};class Person
{Phone* pPhone;
public:Person(){//创建一个手机pPhone = new Phone("华为");}Phone* getP(){return pPhone;}//拷贝 this --> xqPerson(const Person& other)  //const Person& other = xm;{cout<<"copy"<<endl;this->pPhone = other.pPhone;//1.找到小明的指针// other.pPhone 指针 *other.pPhone 指针指向的对象//2.按照小明的指针指向的对象 创建一个一模一样的//new Phone(*other.pPhone)//3.小强的指针 指向新对象this->pPhone = new Phone(*other.pPhone);}
};//1.系统默认的拷贝构造:浅拷贝的过程
//2.浅拷贝:类中包含了一个成员指针,只对成员指针的指向 进行赋值
int main()
{Person xm;Person xq(xm);cout<<xm.getP()<<endl;cout<<xq.getP()<<endl;return 0;
}

 

注意:
      1.系统默认的拷贝构造函数是: 浅拷贝。
      2.当类中有指针类型的成员时,才去考虑深拷贝和浅拷贝的问题。
最后的最后,希望大家点点赞,点点关注,谢谢大家!

 

这篇关于C++中深拷贝与浅拷贝的区别的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/895410

相关文章

深入解析C++ 中std::map内存管理

《深入解析C++中std::map内存管理》文章详解C++std::map内存管理,指出clear()仅删除元素可能不释放底层内存,建议用swap()与空map交换以彻底释放,针对指针类型需手动de... 目录1️、基本清空std::map2️、使用 swap 彻底释放内存3️、map 中存储指针类型的对象

Go之errors.New和fmt.Errorf 的区别小结

《Go之errors.New和fmt.Errorf的区别小结》本文主要介绍了Go之errors.New和fmt.Errorf的区别,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考... 目录error的基本用法1. 获取错误信息2. 在条件判断中使用基本区别1.函数签名2.使用场景详细对

Redis中哨兵机制和集群的区别及说明

《Redis中哨兵机制和集群的区别及说明》Redis哨兵通过主从复制实现高可用,适用于中小规模数据;集群采用分布式分片,支持动态扩展,适合大规模数据,哨兵管理简单但扩展性弱,集群性能更强但架构复杂,根... 目录一、架构设计与节点角色1. 哨兵机制(Sentinel)2. 集群(Cluster)二、数据分片

C++ STL-string类底层实现过程

《C++STL-string类底层实现过程》本文实现了一个简易的string类,涵盖动态数组存储、深拷贝机制、迭代器支持、容量调整、字符串修改、运算符重载等功能,模拟标准string核心特性,重点强... 目录实现框架一、默认成员函数1.默认构造函数2.构造函数3.拷贝构造函数(重点)4.赋值运算符重载函数

C++ vector越界问题的完整解决方案

《C++vector越界问题的完整解决方案》在C++开发中,std::vector作为最常用的动态数组容器,其便捷性与性能优势使其成为处理可变长度数据的首选,然而,数组越界访问始终是威胁程序稳定性的... 目录引言一、vector越界的底层原理与危害1.1 越界访问的本质原因1.2 越界访问的实际危害二、基

一文带你迅速搞懂路由器/交换机/光猫三者概念区别

《一文带你迅速搞懂路由器/交换机/光猫三者概念区别》讨论网络设备时,常提及路由器、交换机及光猫等词汇,日常生活、工作中,这些设备至关重要,居家上网、企业内部沟通乃至互联网冲浪皆无法脱离其影响力,本文将... 当谈论网络设备时,我们常常会听到路由器、交换机和光猫这几个名词。它们是构建现代网络基础设施的关键组成

redis和redission分布式锁原理及区别说明

《redis和redission分布式锁原理及区别说明》文章对比了synchronized、乐观锁、Redis分布式锁及Redission锁的原理与区别,指出在集群环境下synchronized失效,... 目录Redis和redission分布式锁原理及区别1、有的同伴想到了synchronized关键字

c++日志库log4cplus快速入门小结

《c++日志库log4cplus快速入门小结》文章浏览阅读1.1w次,点赞9次,收藏44次。本文介绍Log4cplus,一种适用于C++的线程安全日志记录API,提供灵活的日志管理和配置控制。文章涵盖... 目录简介日志等级配置文件使用关于初始化使用示例总结参考资料简介log4j 用于Java,log4c

C++归并排序代码实现示例代码

《C++归并排序代码实现示例代码》归并排序将待排序数组分成两个子数组,分别对这两个子数组进行排序,然后将排序好的子数组合并,得到排序后的数组,:本文主要介绍C++归并排序代码实现的相关资料,需要的... 目录1 算法核心思想2 代码实现3 算法时间复杂度1 算法核心思想归并排序是一种高效的排序方式,需要用

C++11范围for初始化列表auto decltype详解

《C++11范围for初始化列表autodecltype详解》C++11引入auto类型推导、decltype类型推断、统一列表初始化、范围for循环及智能指针,提升代码简洁性、类型安全与资源管理效... 目录C++11新特性1. 自动类型推导auto1.1 基本语法2. decltype3. 列表初始化3