博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
关于拷贝构造函数和重载成员运算符=
阅读量:2792 次
发布时间:2019-05-13

本文共 2760 字,大约阅读时间需要 9 分钟。

#include 
#include
using namespace std;/********内存重复释放,根本原因就是浅拷贝**********///解决方案1:getName使用引用传递参数,这个时候两个对象其实是同一个地址//解决方案2:自定义拷贝构造函数进行深拷贝//如果浅拷贝且使用按值传递传递参数,则可能会造成野指针/*class Person{public: Person(char *); Person() { } //拷贝构造函数的参数说明了两个问题,1.const限定说明必须做右值;2.使用引用传值为了防止浅拷贝 Person(const Person& p)//括弧中的是右值,而函数如果要传递右值参数一定要用const限定否则编译器不会通过, 这里虽然会通过应该是做过优化了 { char *s = p.m_pName; int l = strlen(s) + 1; m_pName = new char[l]; strcpy(m_pName,s); } ~Person(); Person operator= (const Person& p) { char *s = p.m_pName; int l = strlen(s) + 1; m_pName = new char[l]; strcpy(m_pName,s); return *this; } char *getName(Person); //char *getName(Person&); char *m_pName;};Person::Person(char *s){ int l = strlen(s) + 1; m_pName = new char[l]; strcpy(m_pName,s);}Person::~Person(){ Person *p; p = this; delete []m_pName;}//char * Person::getName(Person &person)//{// return person.m_pName;//}char * Person::getName(Person person){ return person.m_pName;}int main(){ Person me("John"); Person *p1 = &me; Person you("Tom"); Person *p2 = &you; //这个时候必须使用拷贝构造函数和重载运算符=??????????????????? //至于为什么还需要拷贝构造函数,应该是对象之间的赋值需要生成一个副本,而副本的生成如果没有 //使用拷贝构造函数就是简单的位拷贝,在进行堆栈操作的时候会造成野指针。(通过反汇编发现的确是这样做的 ) //而直接初始化或者使用函数返回对象进行初始化的时候只需要拷贝构造函数而不需要重载=,应该是 //编译器优化的结果 Person he; he = me; //需要重载运算符=,因为这个时候已经不是初始化了。 cout << he.m_pName << endl; 若想如此显示赋值,必须同时重载=运算符和使用拷贝构造函数 //Person *he = new Person(); //*he = me; //cout << (*he).m_pName << endl; //delete he; 这样需要使用拷贝构造函数,用一个对象来初始化另外一个对象 //Person he = me; //cout << he.m_pName << endl; //这样的使用方法只能通过拷贝构造函数进行深拷贝,对象按值传递 //me.getName(you); return 0;}*//*************深拷贝和浅拷贝**********************//*拷贝构造函数是一种特殊的构造函数,函数的名称必须和类名称一致,它的唯一的一个参数是本类型的一个引用变量,该参数是const类型,不可变的。例如:类X的拷贝构造函数的形式为X(X& x)。当用一个已初始化过了的自定义类类型对象去初始化另一个新构造的对象的时候,拷贝构造函数就会被自动调用。也就是说,当类的对象需要拷贝时,拷贝构造函数将会被调用。以下情况都会调用拷贝构造函数:一个对象以值传递的方式传入函数体 一个对象以值传递的方式从函数返回 一个对象需要通过另外一个对象进行初始化。 如果在类中没有显式地声明一个拷贝构造函数,那么,编译器将会自动生成一个默认的拷贝构造函数,该构造函数完成对象之间的位拷贝。位拷贝又称浅拷贝自定义拷贝构造函数是一种良好的编程风格,它可以阻止编译器形成默认的拷贝构造函数,提高源码效率*//************************************************************************//*在某些状况下,类内成员变量需要动态开辟堆内存,如果实行位拷贝,也就是把对象里的值完全复制给另一个对象,如A=B。这时,如果B中有一个成员变量指针已经申请了内存,那A中的那个成员变量也指向同一块内存。这就出现了问题:当B把内存释放了(如:析构),这时A内的指针就是野指针了,出现运行错误。深拷贝和浅拷贝可以简单理解为:如果一个类拥有资源,当这个类的对象发生复制过程的时候,资源重新分配,这个过程就是深拷贝,反之,没有重新分配资源,就是浅拷贝。 深拷贝和浅拷贝可以简单理解为:如果一个类拥有资源,当这个类的对象发生复制过程的时候,资源重新分配,这个过程就是深拷贝,反之,没有重新分配资源,就是浅拷贝如果一个类拥有资源(堆,或者是其它系统资源),当这个类的对象发生复制过程的时候,这个过程就可以叫做深拷贝,反之对象存在资源,但复制过程并未复制资源的情况视为浅拷贝。 浅拷贝资源后在释放资源的时候会产生资源归属不清的情况导致程序运行出错*//************************************************************************//*class CA{public: CA(int b,char* cstr) { a=b; str=new char[b]; strcpy(str,cstr); } CA(const CA& C) { a=C.a; str=new char[a]; //深拷贝 if(str!=0) strcpy(str,C.str); } void Show() { cout<
<

转载地址:http://bonmd.baihongyu.com/

你可能感兴趣的文章
决策树之ID3算法
查看>>
朴素贝叶斯算法
查看>>
聚类分析经典算法讲解及实现
查看>>
mongodb中使用分组,聚合和映射-归并
查看>>
使用java连接mongodb数据库,并访问集合
查看>>
java对mongodb数据库的增删改查
查看>>
决策树之ID3算法实现(python)
查看>>
数据预处理
查看>>
大型数据库中的关联规则挖掘
查看>>
Apriori算法
查看>>
linux:shell中>,>>,<的含义
查看>>
远程给局域网中的计算机进行重启或者关机
查看>>
java数据库连接池
查看>>
PicGo + Gitee(码云)实现markdown图床
查看>>
高性能服务器架构(三):分布式缓存
查看>>
PowerEdge R430 机架式服务器安装( Ubuntu server 14.04.1 、PHP5.5.9、PHP-redis2.8、Phalcon3.1)...
查看>>
ngx_lua_API 指令详解(五)coroutine.create,coroutine.resume,coroutine.yield 等集合指令介绍...
查看>>
Nginx模块之Nginx-Ts-Module学习笔记(一)抢险体验
查看>>
Golang入门教程(二)Ubuntu16.04下安装golang(实例:Golang 定时任务管理器)
查看>>
Nginx ab压力测试
查看>>