C++浅拷贝与深拷贝探究


浅拷贝与深拷贝的概念是在类的复制/拷贝构造函数中出现的。

拷贝构造函数使用场景

  • 对象作为参数,以值传递方式传入函数(要调用拷贝构造函数将实参拷贝给函数栈中的形参)
  • 对象作为返回值,以值方式返回(要调用拷贝构造函数,将要传出的对象拷贝给一个外作用域的临时对象)
  • 用一个对象去初始化另外一个对象

浅拷贝:直接使用 “=” 赋值,拷贝给定对象属性。(如果是一般类型,那么就相当于赋值运算符。如果有指针,赋值后就相当于拷贝双方指向同一个地址)

private:

xxx *pStr;

......

pStr = m.pStr;//拷贝的是地址

深拷贝:在赋值之前,先给指针申请一段内存,而后拷贝给定对象属性值。(即:先申请内存,再赋值)

pStr = new xxx[xxx];//先申请内存

对pStr赋值//再拷贝值

实例说明

浅拷贝

 #pragma once
class Demo
{
private:
int m_icount;
int *pStr;
public:
Demo(int icount);//构造函数
Demo(const Demo &d);//拷贝构造函数(浅拷贝)
~Demo();//析构函数
void setValue(int icount);
void printValue();
void printAddr();
};

Demo.h

 #include "Demo.h"
#include <iostream>
using namespace std; Demo::Demo(int icount)
{
m_icount = icount;
pStr = new int[m_icount];
for (int i = ; i < m_icount; i++)
{
pStr[i] = i;
}
cout << "\nDemo(int value)构造函数被调用!" << endl;
} Demo::Demo(const Demo &d)
{
m_icount = d.m_icount;
//浅拷贝
pStr = d.pStr;//浅拷贝,赋地址
cout << "Demo(const Demo &d)浅拷贝构造函数被调用!" << endl; //深拷贝
/*pStr = new int[m_icount];
for (int i = 0; i < m_icount; i++)
{
pStr[i] = d.pStr[i];
}
cout << "\nDemo(const Demo &d)深拷贝构造函数被调用!" << endl;*/
} Demo::~Demo()
{
delete[]pStr;
pStr = NULL;
cout << "\n~Demo()析构函数被调用!" << endl;
} void Demo::setValue(int icount)
{
m_icount = icount;
cout << "\nsetValue(int icount)函数被调用!" << endl;
} void Demo::printValue()
{
for (int i = ; i < m_icount; i++)
{
cout << pStr[i] << " ";
}
cout << "\nDemo::printValue()函数被调用!" << endl;
} void Demo::printAddr()
{
cout << pStr << endl;
} int main()
{
Demo d1();
cout << "\nd1 : ";
d1.printValue(); d1.setValue();
cout << "\nd1 : ";
d1.printValue(); Demo d2(d1);
cout << "\nd2 :";
d2.printValue(); cout << "\nd1 addr : ";
d1.printAddr();
cout << "\nd2 addr : ";
d2.printAddr(); system("pause");
}

Demo.cpp

//代码中的拷贝构造函数形参 const Demo &d

//const: 禁止在传参过程中改变参数对象

//&d:引用传递,参数如果不是引用传递,那么在将实参拷贝给形参时就要调用拷贝构造函数,而函数自身就是拷贝构造函数,就会不断调用自己,陷入递归,无法自拔。->拷贝构造函数必须使用引用传递!!!

结果

那么,对于两者中任意一个对象进行操作就势必会改变另外一个,这是我们一般不希望看到的。这时候,我们就需要深拷贝。为即将产生的新对象重新申请存储空间。

深拷贝

 #include "Demo.h"
#include <iostream>
using namespace std; Demo::Demo(int icount)
{
m_icount = icount;
pStr = new int[m_icount];
for (int i = ; i < m_icount; i++)
{
pStr[i] = i;
}
cout << "\nDemo(int value)构造函数被调用!" << endl;
} Demo::Demo(const Demo &d)
{
m_icount = d.m_icount;
//浅拷贝
/*pStr = d.pStr;//浅拷贝,赋地址
cout << "Demo(const Demo &d)浅拷贝构造函数被调用!" << endl;*/ //深拷贝
pStr = new int[m_icount];
for (int i = ; i < m_icount; i++)
{
pStr[i] = d.pStr[i];
}
cout << "\nDemo(const Demo &d)深拷贝构造函数被调用!" << endl;
} Demo::~Demo()
{
delete[]pStr;
pStr = NULL;
cout << "\n~Demo()析构函数被调用!" << endl;
} void Demo::setValue(int icount)
{
m_icount = icount;
cout << "\nsetValue(int icount)函数被调用!" << endl;
} void Demo::printValue()
{
for (int i = ; i < m_icount; i++)
{
cout << pStr[i] << " ";
}
cout << "\nDemo::printValue()函数被调用!" << endl;
} void Demo::printAddr()
{
cout << pStr << endl;
} int main()
{
Demo d1();
cout << "\nd1 : ";
d1.printValue(); d1.setValue();
cout << "\nd1 : ";
d1.printValue(); Demo d2(d1);
cout << "\nd2 :";
d2.printValue(); cout << "\nd1 addr : ";
d1.printAddr();
cout << "\nd2 addr : ";
d2.printAddr(); system("pause");
}

Demo.cpp

结果

简言之,浅拷贝就是把别人的地址拿过来作为自己的,双方共同维护一个对象,“绑在一块,一损俱损”;而深拷贝则是赋值之前为新对象申请了额外的空间,所以独立于被拷贝对象,“咱没关系,爱咋咋地”!

拷贝构造函数使用原则

  • 含有指针类型的成员或者含有动态内存空间的对象,必须提供自定义的拷贝构造函数。
  • 在提供拷贝构造函数的同时,也要考虑实现自定义的赋值运算符。

怎样区别

赋值运算还是拷贝函数?有无新对象产生

深拷贝还是浅拷贝?有无指针成员和动态内存空间

C++ 浅拷贝与深拷贝探究的更多相关文章

  1. 【转】Python中的赋值、浅拷贝、深拷贝介绍

    这篇文章主要介绍了Python中的赋值.浅拷贝.深拷贝介绍,Python中也分为简单赋值.浅拷贝.深拷贝这几种"拷贝"方式,需要的朋友可以参考下   和很多语言一样,Python中 ...

  2. [转] js对象浅拷贝和深拷贝详解

    本文为大家分享了JavaScript对象的浅拷贝和深拷贝代码,供大家参考,具体内容如下 1.浅拷贝 拷贝就是把父对像的属性,全部拷贝给子对象. 下面这个函数,就是在做拷贝: var Chinese = ...

  3. 渐析java的浅拷贝和深拷贝

          首先来看看浅拷贝和深拷贝的定义:       浅拷贝:使用一个已知实例对新创建实例的成员变量逐个赋值,这个方式被称为浅拷贝.       深拷贝:当一个类的拷贝构造方法,不仅要复制对象的所 ...

  4. 关于JavaScript的浅拷贝和深拷贝

    在 JS 中有一些基本类型像是Number.String.Boolean,而对象就是像这样的东西{ name: 'Larry', skill: 'Node.js' },对象跟基本类型最大的不同就在于他 ...

  5. c#中浅拷贝和深拷贝的理解

    c#中拷贝有浅拷贝和深拷贝之分. 例如对象A,其中有值类型字段和引用类型字段: 1.浅拷贝: 对于值类型字段,直接逐位复制到新拷贝的副本对象中,修改副本的字段的值,不会影响源对象中字段的值: 对于引用 ...

  6. C#浅拷贝与深拷贝区别

    也许会有人这样解释C# 中浅拷贝与深拷贝区别: 浅拷贝是对引用类型拷贝地址,对值类型直接进行拷贝. 不能说它完全错误,但至少还不够严谨.比如:string 类型咋说? 其实,我们可以通过实践来寻找答案 ...

  7. IOS的浅拷贝和深拷贝

    什么是深拷贝和浅拷贝 浅拷贝:就是指针的复制,拷贝的指针跟原指针指向内存中的同一个位置的对象.至于对象的引用计数值是否+1,就是看拷贝的指针赋给给的变量是Strong类型的,还是week类型的. 如果 ...

  8. 关于python中赋值、浅拷贝、深拷贝之间区别的深入分析

    当重新学习了计算机基础课程<数据结构和算法分析>后再来看这篇自己以前写的博文,发现错误百出.python内置数据类型之所以会有这些特性,归根结底是它采用的是传递内存地址的方式,而不是传递真 ...

  9. js中的浅拷贝和深拷贝

    说说最近所学:浅拷贝和深拷贝也叫做浅克隆和深克隆,深浅主要针对的是对象的"深度",常见的对象都是"浅"的,也就是对象里的属性就是单个的属性,而"深&q ...

随机推荐

  1. Linux中搭建Maven私服

    linux安装maven 先解压maven的压缩包apache-maven-3.5.4-bin.tar.gz   命令: tar -zavf pache-maven-3.5.4-bin.tar.gz ...

  2. [Swift]LeetCode206. 反转链表 | Reverse Linked List

    Reverse a singly linked list. Example: Input: 1->2->3->4->5->NULL Output: 5->4-> ...

  3. [Swift]LeetCode341. 压平嵌套链表迭代器 | Flatten Nested List Iterator

    Given a nested list of integers, implement an iterator to flatten it. Each element is either an inte ...

  4. zuul进阶学习(二)

    1. zuul进阶学习(二) 1.1. zuul对接apollo 1.1.1. Netflix Archaius 1.1.2. 定期拉 1.2. zuul生产管理实践 1.2.1. zuul网关参考部 ...

  5. awk小例子_2_数值统计脚本

    通信公司工作,经常处理各种协议接口,在统计协议接口字段内容时,需要统计字段填写的内容是否正确,和占比是多少.要是单次统计,估计会把人累死,写个脚本统计,轻松便捷. 举例:接口内容 这是一条话单,这样的 ...

  6. Java中的方法(形参及实参)return返回类型

    如何定义 Java 中的方法 所谓方法,就是用来解决一类问题的代码的有序组合,是一个功能模块. 一般情况下,定义一个方法的语法是: 其中: 1. 访问修饰符:方法允许被访问的权限范围, 可以是 pub ...

  7. socket编程: TypeError: must be bytes or buffer, not str

    先看一段代码 #!/usr/bin/env python3 from socket import * serverName = "10.10.10.132" serverPort ...

  8. JDK1.8 ConcurrentHashMap源码阅读

    1.  带着问题去阅读 为什么说ConcurrentHashMap是线程安全的?或者说 ConcurrentHashMap是如何防止并发的? 2.  字段和常量 首先,来看一下ConcurrentHa ...

  9. 带着萌新看springboot源码8(spring ioc源码 完)

    上一节说到实例化了所有的单实例Bean,后面还有一步遍历 12.完成容器刷新(finishRefresh();) 那个和生命周期有关的后置处理器类型是LifecycleProcessor:监听器原理我 ...

  10. Python进阶:自定义对象实现切片功能

    2018-12-31 更新声明:切片系列文章本是分三篇写成,现已合并成一篇.合并后,修正了一些严重的错误(如自定义序列切片的部分),还对行文结构与章节衔接做了大量改动.原系列的单篇就不删除了,毕竟也是 ...