c++浅拷贝和深拷贝---14
原创博文,转载请标明出处--周学伟http://www.cnblogs.com/zxouxuewei/
1.什么是拷贝构造函数:
2.调用拷贝构造函数的情形:
(1)一个对象以值传递的方式传入函数体
#include <iostream>
#include <cstring> using namespace std; class test
{
public:
test();
test(const test& ctest);
~test();
private:
char Name;
}; test::test()
{
cout <<"Construction "<<endl;
} test::test(const test& ctest)
{
cout <<"Copy Construction "<<endl;
} test::~test()
{
cout <<"Destruction "<<endl;
} void function(test param_test)
{
cout <<"into function"<<endl;
} int main(int argc, char const *argv[])
{
test Atest;
function(Atest); return 0;
}
运行结果
(2)一个对象以值传递的方式从函数返回
#include <iostream>
#include <cstring> using namespace std; class test
{
public:
test();
test(const test& ctest);
~test();
private:
char Name;
}; test::test()
{
cout <<"Construction "<<endl;
} test::test(const test& ctest)
{
cout <<"Copy Construction "<<endl;
} test::~test()
{
cout <<"Destruction "<<endl;
} test function(test param_test)
{
cout <<"into function"<<endl;
return param_test;
} int main(int argc, char const *argv[])
{
test Atest;
function(Atest); return 0;
}
运行结果:
(3)一个对象需要通过另外一个对象进行初始化。
#include <iostream>
#include <cstring> using namespace std; class test
{
public:
test();
test(const test& ctest);
~test();
private:
char Name;
}; test::test()
{
cout <<"Construction "<<endl;
} test::test(const test& ctest)
{
cout <<"Copy Construction "<<endl;
} test::~test()
{
cout <<"Destruction "<<endl;
} int main(int argc, char const *argv[])
{
test Atest;
test Btest = Atest; return 0;
}
运行结果:
3.浅拷贝和深拷贝:
在某些状况下,类内成员变量需要动态开辟堆内存,如果实行位拷贝,也就是把对象里的值完全复制给另一个对象,如A=B。这时,如果B中有一个成员变量指针已经申请了内存,那A中的那个成员变量也指向同一块内存。这就出现了问题:当B把内存释放了(如:析构),这时A内的指针就是野指针了,出现运行错误。深拷贝和浅拷贝可以简单理解为:如果一个类拥有资源,当这个类的对象发生复制过程的时候,资源重新分配,这个过程就是深拷贝,反之,没有重新分配资源,就是浅拷贝。
---》深拷贝:(查看在堆上分配的内存资源进行深拷贝)
#include <iostream>
#include <cstring> using namespace std; class test
{
public:
test(char* name);
test(const test& ctest);
~test();
void Printf();
private:
char* PName;
}; test::test(char* name)//普通构造函数
{
if(name != NULL)
{
cout <<"Construction "<<name<<endl;
int len = strlen(name)+1;
PName = new char[len]; memset(PName, 0, len);
strcpy(PName,name);
}
else
{
PName = NULL;
}
cout <<"&PName = "<<static_cast<void*>(PName)<<endl;
} test::test(const test& ctest)//深拷贝构造函数
{
if(ctest.PName != NULL)
{
cout <<"Copy Construction "<<endl;
int len = strlen(ctest.PName)+1;
PName = new char[len];
memset(PName, 0, len);
strcpy(PName,ctest.PName);
}
else
{
PName = NULL;
}
cout <<"&PName = "<<static_cast<void*>(PName)<<endl;
} test::~test()
{
cout <<"Destruction "<<endl;
if(PName != NULL)
{
Printf();
delete []PName;
PName = NULL;
}
} void test::Printf()
{
cout <<"&PName = "<<static_cast<void*>(PName)<<endl;
} int main(int argc, char const *argv[])
{
test Atest((char*)"zhouxuewei"); test Btest = Atest;//复制对象的副本,此刻调用拷贝构造函数 return 0;
}
运行结果:
----》浅拷贝(使用编译器默认的拷贝构造函数)
#include <iostream>
#include <cstring> using namespace std; class test
{
public:
test(char* name);
//test(const test& ctest);
~test();
void Printf();
private:
char* PName;
}; test::test(char* name)//普通构造函数
{
if(name != NULL)
{
cout <<"Construction "<<name<<endl;
int len = strlen(name)+;
PName = new char[len]; memset(PName, , len);
strcpy(PName,name);
}
else
{
PName = NULL;
}
cout <<"&PName = "<<static_cast<void*>(PName)<<endl;
}
/*
test::test(const test& ctest)//深拷贝构造函数
{
if(ctest.PName != NULL)
{
cout <<"Copy Construction "<<endl;
int len = strlen(ctest.PName)+1;
PName = new char[len];
memset(PName, 0, len);
strcpy(PName,ctest.PName);
}
else
{
PName = NULL;
}
cout <<"&PName = "<<static_cast<void*>(PName)<<endl;
}*/ test::~test()
{
cout <<"Destruction "<<endl;
if(PName != NULL)
{
Printf();
delete []PName;
PName = NULL;
}
} void test::Printf()
{
cout <<"&PName = "<<static_cast<void*>(PName)<<endl;
} int main(int argc, char const *argv[])
{
test Atest((char*)"zhouxuewei"); test Btest = Atest;//复制对象的副本,此刻调用拷贝构造函数 return ;
}
运行结果:结果可以看出,拷贝的对象使用的堆内存地址相同,析构时free两次,产生错误:
4.赋值运算符
#include <iostream>
#include <cstring> using namespace std; class test
{
public:
test(char* name,int value);
test(const test& ctest);
~test();
test& operator=(const test& otest);
void Printf();
private:
char* PName;
int value;
static unsigned char count;
}; unsigned char test::count = 0; test::test(char* name,int val)//普通构造函数
{
if(name != NULL)
{
int len = strlen(name)+1;
PName = new char[len]; memset(PName, 0, len);
strcpy(PName,name);
}
else
{
PName = NULL;
}
value = val;
cout <<"&PName = "<<static_cast<void*>(PName)<<", name = "<<PName<<", value = "<<value<<endl;
count++;
} test::test(const test& ctest)//深拷贝构造函数
{
if(ctest.PName != NULL)
{
cout <<"Copy Construction "<<endl;
int len = strlen(ctest.PName)+1;
PName = new char[len];
memset(PName, 0, len);
strcpy(PName,ctest.PName);
}
else
{
PName = NULL;
}
cout <<"&PName = "<<static_cast<void*>(PName)<<", name = "<<PName<<endl;
count++;
} test::~test()
{
cout <<"Destruction "<<", count = "<<static_cast<int>(count)<<endl;
if(PName != NULL)
{
Printf();
delete []PName;
PName = NULL;
}
count--;
} test& test::operator=(const test& otest)
{
cout <<"into operator "<<endl;
if(&otest == this) //自赋值检查
{
return *this;
}
if(PName != NULL)
{
delete []PName; //原有资源的释放
PName = NULL;
}
if(otest.PName != NULL)
{
int len = strlen(otest.PName)+1;//开始赋值
PName = new char[len];
memset(PName, 0, len);
strcpy(PName, otest.PName);
}
else
PName = NULL;
value = otest.value;
return *this;
} void test::Printf()
{
cout <<"&PName = "<<static_cast<void*>(PName)<<",name = "<<PName<<", value = "<<value<<endl;
} int main(int argc, char const *argv[])
{
test Atest((char*)"zhouxuewei",111);
test Btest((char*)"wanghuixi",222); Atest = Btest; return 0;
}
运行结果:
总结:
1》何时需要定义拷贝构造函数:
a.类数据成员有指针。
b.类数据成员管理资源(如打开文件)
2》如果一个类需要析构函数来释放资源,则他需要一个拷贝构造函数。
3》如果想禁止一个类的拷贝构造,则需要将拷贝构造函数声明为private。
c++浅拷贝和深拷贝---14的更多相关文章
- 编写高质量代码改善C#程序的157个建议——建议14: 正确实现浅拷贝和深拷贝
建议14: 正确实现浅拷贝和深拷贝 为对象创建副本的技术称为拷贝(也叫克隆).我们将拷贝分为浅拷贝和深拷贝. 浅拷贝 将对象中的所有字段复制到新的对象(副本)中.其中,值类型字段的值被复制到副本中后, ...
- 编写高质量代码改善C#程序的157个建议[为类型输出格式化字符串、实现浅拷贝和深拷贝、用dynamic来优化反射]
前言 本文已更新至http://www.cnblogs.com/aehyok/p/3624579.html .本文主要学习记录以下内容: 建议13.为类型输出格式化字符串 建议14.正确实现浅拷贝和深 ...
- 浅析JavaScript解析赋值、浅拷贝和深拷贝的区别
文章首发于sau交流学习社区 一.赋值(Copy) 赋值是将某一数值或对象赋给某个变量的过程,分为: 1.基本数据类型:赋值,赋值之后两个变量互不影响 2.引用数据类型:赋**址**,两个变量具有相同 ...
- js对象浅拷贝和深拷贝详解
js对象浅拷贝和深拷贝详解 作者:i10630226 字体:[增加 减小] 类型:转载 时间:2016-09-05我要评论 这篇文章主要为大家详细介绍了JavaScript对象的浅拷贝和深拷贝代码,具 ...
- C#程序编写高质量代码改善的157个建议【13-15】[为类型输出格式化字符串、实现浅拷贝和深拷贝、用dynamic来优化反射]
前言 本文已更新至http://www.cnblogs.com/aehyok/p/3624579.html .本文主要学习记录以下内容: 建议13.为类型输出格式化字符串 建议14.正确实现浅拷贝和深 ...
- java中的浅拷贝和深拷贝
复制 将一个对象的引用复制给另一个对象,一共有三种方式.第一种方式是直接赋值,第二种方式是浅复制,第三种方式是深复制. 1.直接赋值 在Java中,A a1 = a2,这实际上复制的是引用,也就是说 ...
- [转] js对象浅拷贝和深拷贝详解
本文为大家分享了JavaScript对象的浅拷贝和深拷贝代码,供大家参考,具体内容如下 1.浅拷贝 拷贝就是把父对像的属性,全部拷贝给子对象. 下面这个函数,就是在做拷贝: var Chinese = ...
- 【转】Python中的赋值、浅拷贝、深拷贝介绍
这篇文章主要介绍了Python中的赋值.浅拷贝.深拷贝介绍,Python中也分为简单赋值.浅拷贝.深拷贝这几种"拷贝"方式,需要的朋友可以参考下 和很多语言一样,Python中 ...
- 渐析java的浅拷贝和深拷贝
首先来看看浅拷贝和深拷贝的定义: 浅拷贝:使用一个已知实例对新创建实例的成员变量逐个赋值,这个方式被称为浅拷贝. 深拷贝:当一个类的拷贝构造方法,不仅要复制对象的所 ...
随机推荐
- ML-DL-各种资源汇总
1.Used Libraries, Datasets, and Models 1.1 Libraries TensorFlow (from Google): https://www.tensorflo ...
- 使用FineReport打造考试分析系统
本系统的优点: 1.报表内容丰富:系统中包含总分分析.小分分析.作答错因分析.试卷命题分析和各类用户报告单五类报表.涵盖学校须要的各项分析数据,并提供丰富的图表,使分析数据更直观表现. 2.操作灵活简 ...
- windows下安装和配置多个版本的JDK
https://jingyan.baidu.com/article/47a29f2474ba55c015239957.html 如何在windows下安装和配置多个版本的jdk,本文将带你在windo ...
- Mybatis if标签判断大小
1.if标签语法 <select...> SQL语句1 <if test="条件表达式"> SQL语句2 </if> </select&g ...
- Struts2、Spring、Hibernate 高效开发的最佳实践(转载)
Struts2.Spring.Hibernate 高效开发的最佳实践 Struts2.Spring.Hibernate(SSH)是最常用的 Java EE Web 组件层的开发技术搭配,网络中和许多 ...
- 关于安装black apple遇到的坑
1.用料准备: vmware player或vmware workstation,osx系统安装包(后缀名为cdr的几个G大小的文件),为vmware安装osx的补丁安装包unlocker 2.在安装 ...
- String直接赋值和使用new的区别
String str1 = "ABC"; String str2 = new String("ABC"); String str1 = “ABC”;可能创建一个 ...
- Windows IOT 开发入门(硬件入门)
接上文,在准备工作完成之后.接下来应该要熟悉硬件和架构了. 以下是一个简易物联网架构设计图 关于微软云这里就不说太多了.有兴趣的朋友可以去这里了解更多https://www.azure.cn/. 在上 ...
- ubuntu 挂载硬盘
https://cndaqiang.github.io/2017/10/11/ubuntu-disk/ 查看硬盘 查看方法一 查看/dev下面的设备文件 ll -h /dev/sd* 通过sudo f ...
- python进行数据分析------相关分析
相关分析 import statsmodels.api as sm import pandas as pd import numpy as np from patsy.highlevel import ...