浅谈c++中的复制构造函数

3. 复制构造函数

3.1 复制构造函数作用

它是一种特殊的构造函数,其形参为本类对象的引用。作用是用一个已存在的对象去初始化同类型的新对象。

3.2 声明和实现复制构造函数的一般方法

Class Clock{

Public:

Clock(int newH,int newM,int newS);//构造函数

Clock(const Clock &a); //复制构造函数声明

...........

Private:

Int x,y;

}

Clock::Clock(const Clock &a){ // 复制构造函数的实现。

x=a.x;

y=a.y;

}

3.3 调用复制构造函数的三种情况

(1)定义一个对象时,以本类另一个对象作为初始值,发生复制构造;

(2)如果函数的形参是类的对象,调用函数时,将使用实参对其初始化,发生复制构造。

(3)如果函数的返回值是类的对象,函数执行完成返回主调函数时,将使用return语句中的对象初始化一个临时无名对象,传递给主调函数,此时发生复制构造。

例如:

Class Point{

Public:

Point(int xx=0, int yy=0){x=xx;y=yy;}//构造函数

Point(const Point &P);//复制构造函数

..........

}

//形参为Point类对象的函数

Void fun1(point P){

Cout<<p.getX()<<endl;

}

//返回值为point类对象的函数

Point fun2(){

Point a(1,2);

return a;

}

//主程序

int main(){

Point a(4,5); //第一个对象A

Point b(a); //情况一,用A初始化B。第一次调用复制构造函数

cout<<b.getX()<<endl;

fun1(b); // 情况二,对象B作为fun1的实参。第二次调用复制构造函数

b=fun2(); //情况三,函数的返回值是类的对象,函数返回时调用复制构造函

cout<< b.getX()<<endl;

return 0;

}

3.4 隐含的复制构造函数

如果程序员没有为类声明复制构造函数,则编译器自己生成一个隐含的复制构造函数。

这个隐含的复制构造函数的作用:为初始值的对象的每个数据成员的值,初始化将要建立对象的数据成员。

3.5 为什么要复制构造函数

肯定不会原样复制,可能复制对象和原对象并不一样的。不是一一对应复制,可能有偏移。这样的话默认的复制构造函数就没用了,自己来声明复制构造函数。这里考虑深层复制构造函数。

3.6浅拷贝缺点

(1)浅拷贝只是复制了指针,也就是说两个指针指向同一块内存,通过一个指针访问这块内存并进行改变,那么用第二个指针可以观察到对象是变化的。所以浅拷贝后这两个对象不是独立的。

(2)浅拷贝后,对对象进行析构,会对一块内存析构两次,所以会产生错误。

3.6深拷贝怎么实现

指针拷贝的同时,也要拷贝指向的内存区的数据。做到两块内存相互独立。

例子:

Arrayofpoints::Arrayofpoints(const Arrryofpoints & v){

Size = v.size; //参数对象的size复制到当前对象的size中

Points = new Point[size];//首地址不能直接复制,先分配内存空间,构建新数

//组。

for(int i = 0;i<size;i++)

Points[i] = v.points[i];//把原来数组中的元素复制过来。

}

时间: 2024-11-13 06:39:44

浅谈c++中的复制构造函数的相关文章

浅谈C#中new、override、virtual关键字的区别

OO思想现在已经在软件开发项目中广泛应用,其中最重要的一个特性就是继承,最近偶简单的学习了下在设计模式中涉及到继承这个特性时,所需要用到的关键字,其中有一些关键点,特地整理出来. 一.New 在C#中,new这个关键字使用频率非常高,主要有3个功能: a)   作为运算符用来创建一个对象和调用构造函数. b)   作为修饰符. c)   用于在泛型声明中约束可能用作类型参数的参数的类型. 在本文中,只具体介绍new作为修饰符的作用,在用作修饰符时,new关键字可以在派生类中隐藏基类的方法,也就说

浅谈Java中的深拷贝和浅拷贝

浅谈Java中的深拷贝和浅拷贝(转载) 原文链接: http://blog.csdn.net/tounaobun/article/details/8491392 假如说你想复制一个简单变量.很简单: [java] view plaincopyprint? int apples = 5; int pears = apples; int apples = 5; int pears = apples; 不仅仅是int类型,其它七种原始数据类型(boolean,char,byte,short,float

浅谈C#中的常量、类型推断和作用域

浅谈C#中的常量.类型推断和作用域 作者: 字体:[增加 减小] 类型:转载 时间:2013-12-19我要评论 这篇文章主要介绍了C#中的常量.类型推断和作用域,有需要的朋友可以参考一下 一.常量常量是其值在使用过程中不会发生变化的变量.在声明和初始化变量时,在变量前面家关键字const,就可以把该变量指定为一个常量: const int a=100;//a的值将不可以改变 常量的特征: 1.常量必须在声明时初始化.指定了其值以后,就不能再修改了.2.常量的值必须能在编译时用于计算.因此不能从

浅谈js中的this关键字

浅谈js中的this关键字 全局作用域中的this 函数作用域中的this 不同函数调用方法下的this 直接调用 作为对象的方法调用 作为构造函数调用 通过call或apply方法调用 嵌套函数作用域中的this 浅谈js中的this关键字 this是JavaScript中的关键字之一,在编写程序的时候经常会用到,正确的理解和使用关键字this尤为重要.接下来,笔者就从作用域的角度粗谈下自己对this关键字的理解,希望能给到大家一些启示,权当交流之用. 全局作用域中的this 本文将以作用域由

浅谈JavaScript中的原型模式

在JavaScript中创建对象由很多种方式,如工厂模式.构造函数模式.原型模式等: <pre name="code" class="html">//工厂模式 function createPerson(name,age,job) { var o = new Object; o.name = name; o.age = age; o.job = job; o.sayName = function() { alert(this.name); } retur

Java网络编程和NIO详解7:浅谈 Linux 中NIO Selector 的实现原理

Java网络编程和NIO详解7:浅谈 Linux 中NIO Selector 的实现原理 转自:https://www.jianshu.com/p/2b71ea919d49 本系列文章首发于我的个人博客:https://h2pl.github.io/ 欢迎阅览我的CSDN专栏:Java网络编程和NIO https://blog.csdn.net/column/details/21963.html 部分代码会放在我的的Github:https://github.com/h2pl/ 浅谈 Linux

浅谈数据库系统中的cache(转)

http://www.cnblogs.com/benshan/archive/2013/05/26/3099719.html 浅谈数据库系统中的cache(转) Cache和Buffer是两个不同的概念,简单的说,Cache是加速"读",而buffer是缓冲"写",前者解决读的问题,保存从磁盘上读出 的数据,后者是解决写的问题,保存即将要写入到磁盘上的数据.在很多情况下,这两个名词并没有严格区分,常常把读写混合类型称为buffer cache,本文后续的论述中,统一

【转】浅谈Java中的equals和==

浅谈Java中的equals和== 在初学Java时,可能会经常碰到下面的代码: 1 String str1 = new String("hello"); 2 String str2 = new String("hello"); 3 4 System.out.println(str1==str2); 5 System.out.println(str1.equals(str2)); 为什么第4行和第5行的输出结果不一样?==和equals方法之间的区别是什么?如果在初

浅谈oracle中rowid和rownum

[ 概要 ] 刚刚接触oracle的同学可能常常会被rowid和rownum这两个词弄混, 弄清楚这两个家伙对于我们写sql会有很大的帮助, 下面偶就抛砖引玉, 简单地谈谈他们之间的区别吧. [ 比较 ] rowid和rownum都是oracle中的伪列, 但他们还是存在本质区别: rowid: 是物理地址, 用于定位数据表中数据的位置, 它是唯一的且不会改变. rownum: 是根据查询的结果集给每行分配的一个逻辑编号, 查询结果不同, rownum自然不同. 对于同一条记录, 查询条件不同,