分享
 
 
 

复制构造函数

王朝百科·作者佚名  2010-06-18
窄屏简体版  字體: |||超大  

复制构造函数,直接初始化,复制初始化,赋值,临时对象

复制构造函数应弄清的几个问题:何时调用复制构造函数,复制构造函数有何功能,为什么要定义自已的复制构造函数。

1.复制构造函数:当用户没有定义自已的复制构造函数时系统将生成一个默认的复制构造函数。当按值传递对象时,就会创建一个形参的临时对象,然后调用复制构造函数把临时对象的值复制给实参。

2.默认复制构造函数的功能:将一个对象的非静态成员的值逐个复制给另一个对象,注意复制的是成员的值,这种复制方式也称为浅复制。因为静态成员属于整个类,而不属于某个对象,所以调用复制构造函数时静态成员不会受到影响。

3.何时生成临时对象:情形1:按值传递对象注意是按值传递对象,按值传递意味着会创建一个原始对象的副本,

情形2:函数反回对象时。

情形3:用一个对象初始化另一个对象时即复制初始化,语句hyong x=y和hyong x=hyong(y)这里y是hyong类型的对象。都将调用复制构造函数,但有可能创建临时对象也有可能不创建临时对象而用复制构造函数直接初始化对象,这取决于编译器。

4.临时对象是由复制构造函数创建的,当临时对象消失时会调用相应的析构函数。也就是说只要创建了临时对象就会多调用一次析构函数。

5.何时使用复制构造函数:按值传递对象,函数反回对象,用一个对象初始化另一个对象即复制初始化时,根据元素初始化列表初始化数组元素。这四种情况都将调用复制构造函数。记住,复制构造函数只能用于初始化,不能用于赋值,赋值时不会调用复制构造函数,而是使用赋值操作符。

6.直接初始化:直接初始化是把初始化式放在圆括号中的,对于类类型来说,直接初始化总是调用与实参匹配的构造函数来初始化的,

7.复制初始化与复制构造函数:复制初始化使用=等于符号来初始化,复制初始化也是创建一个新对象,并且其初值来自于另一个已存在的对象,复制初始化总是调用复制构造函数来初始化的,复制初始化时首先使用指定的构造函数创建一个临时对象,然后用复制构造函数将临时对象的每个非static成员依次的复制到新创建的对象。复制构造函数执行的是逐个成员初始化。注意这里是用一个已存在的对象创建另一个新对象,与用构造函数直接创建一个新对象不一样,使用构造函数初始化时不会使用另一个对象。比如有类hyong,则语句hyong m(1,2)调用构造函数直接初始化,而语句hyong n=m则是用已存在的对象m去初始化一个新对象n,属于复制初始化。

8.理解赋值与复制初始化的区别(重点):赋值是在两个已存在的对象间进行的,也就是用一个已存在的对象去改变另一个已存在对象的值。赋值将调用赋值操作符对对象进行操作,赋值操作符将在操作符重载中讲解。比如有类hyong,有语句hyong x(1);hyong y(1,2)则x=y;这就是赋值,因为对象x和y是已经存在的对象,而语句hyong x=y;则是复制初始化,是用一个已存在的对象y去创建一个新对象x,所以是复制初始化。

9.复制初始化和赋值是在两个对象之间进行的操作,而直接初始化则不是。

10.注意:使用复制构造函数不一定创建临时对象就如语句hyong x=hyong(y),其中y是hyong类型的对象,就有可能不创建临时对象,这取决于编译器。这里如果创建了临时对象则当临时对象消亡时将调用一次析构函数,而如果没有调用而是直接用复制构造函数初始化对象的就不会调用析构函数。

11.复制构造函数的形式:hyong(const hyong & obj);它接受一个指向类对象的常量引用作为参数。定义为const是必须的,因为复制构造函数只是复制对象,所以没必要改变传递来的对象的值,声明为引用可以节省时间,如果是按值传递的话就会生成对象的副本,会浪费资源,而引用就不会。

12.为什么需要定义自已的复制构造函数:如果类只包含类类型成员和内置类型的成员,则可以不用显示定义复制构造函数。但如果类中包含有指针或者有分配其他类型资源时就必须重新定义复制构造函数。因为类中有指针成员,当把用一个对象初始化另一个对象时,这时两个对象中的指针都指向同一段内存,这时如果其中一个对象被消毁了,这时对象中指针所指向的内存也同样被消毁,但另一个对象确不知道这种情况,这时就会出现问题。比如hyong类中含有一个成员指针p,当声明了hyong x=y其中y也是hyong类的对象,这时对象x和y中的指针成员p都指向同一段内存,而如果y被消毁,但x还没被消毁时就会出问题,这时y中对象的成员指针p已经释放了该内存资源,而x中的成员指针p还不知道已经释放了该资源,这时就会出问题。因为对象x和y中的成员指针共享同一段内存,所以对y中的成员指针p的修改就会影响到对象x中的成员指针。所有这些情况都需要重定义复制构造函数来显示的初始化成员的值,这种初始化方式也被称为深度复制。

13.如果显示定义了复制构造函数则调用显示复制构造函数来直接初始化对象,如果没有显示定义复制构造函数,则调用默认的复制构造函数直接初始化对象。

14.注意:1.在VC++中语句hyong n=m不生成临时对象,但如果显示定义了复制构造函数则调用显示复制构造函数来直接初始化对象n,如果没有显示定义复制构造函数,则调用默认的复制构造函数直接初始化对象n。

2.在VC++中语句hyong m1=hyong(m)有可能生成临时对象也有可能不生成临时对象,如果显示定义了复制构造函数则用复制构造函数直接初始化对象m1,不生成临时对象。如果没有显示定义复制构造函数则复制构造函数将创造临时对象,初始化对象m1

15.C++自动提供的成员函数,有:默认构造函数,复制构造函数,默认析构函数,赋值操作符,地址操作符即this指针,这五种函数如果用户没有定义,则系统会自动创建一个。

16.直接调用类中的构造函数:可以在类中的函数,类外的独立函数,即main()函数中直接调用某一个类的构造函数,比如在main函数中可以有语句n=A(4);这里n是类A的对象,这里就是直接调用类A的构造函数创建一个类A的临时对象,然后把该临时对象的值赋给类A的对象n。在类中的函数和在类外的函数调用类的构造函数的方法和这里类似。注意语句n.A(4)是错误的语句,不能由对象调用类中的构造函数。

例:复制构造函数的使用

class hyong

{public: int a,b,c; hyong(){a=b=c=0;cout<<"gouchao"<<"

";} hyong(int i){a=b=c=i;cout<<"gouchao2"<<"

";}

~hyong(){cout<<"xigou"<<"

";} hyong(const hyong &obj){a=b=c=9;cout<<"fuzi"<<"

";} //复制构造函数。};

void h(hyong k){cout<<"haoshu"<<k.a<<k.b<<"

";} //按值传递对象

hyong f(){hyong m3(5); return m3;} //反回对象。

//如果显示定义了复制构造函数则调用显示复制构造函数来直接初始化对象,如果没有显示定义复制构造函数,则调用默认的复制构造函数直接初始化对象。

int main()

{//以下为几种复制初始化的方式。

hyong m(1);

// hyong n=m和hyong m1=hyong(m)是否生成临时对象依编译器而定

hyong n=m; //在VC++中此语句不生成临时对象,调用显示定义的复制构造函数初始化对象

cout<<m.a<<m.b<<"

";//输出99

cout<<n.a<<n.b<<"

";//,输出99,调用显示定义的复制构造函数初始化对象n,而不会生成临时对象。

hyong m1=hyong(m); //此语句要特别注意,因为此语句有可能生成临时对象也有可能不生成临时对象,如果显示定义了复制构造函数则用复制构造函数直接初始化对象m1,而不会生成临时对象。如果没有显示定义复制构造函数则复制构造函数将生成临时对象,然后对m1进行初始化。

cout<<m1.a<<m1.b<<"

"; //输出输出11,调用显示定义的复制构造函数初始化对象m1,不生成临时对象。如果没有定义复制构造函数则输出,同时会生成临时对象,临时对象撤消时会调用析构函数。

hyong m2(m); cout<<m2.a<<m2.b<<"

"; //输出11,直接调用复制构造函数,因此不会生成临时对象

hyong *p=new hyong(m); cout<<p->a<<p->b<<"

"; //不生成临时对象,直接调用复制构造函数初始化。

//按值传递和反回对象的例子。

h(m); cout<<"kkk"<<"

"; //按值传递对象m,当调用函数h时就会使用复制控制函数生成一个临时对象,然后把这个临时对象复制给实参,当函数调用完毕时就会撤消临时对象,此时会调用一个析构函数,析构函数在h函数的作用域消失时才调用,也就是说在执行了h的函数体后才会调用析构函数。

hyong m4=f(); cout<<m4.a<<m4.b<<"

"; //输出55,用返回的对象初始化对象m4,此语句没有生成临时对象,原因还不清楚,待考证,可能与语句是复制初始化有关。

hyong m5; m5=f(); //此语句调用函数f,f反回一个对象,在反回时会调用复制构造函数生成一个临时对象,并把这个临时对象作为默认赋值操作符的一个参数,因此这里不但调用了复制构造函数还调用了赋值操作符。

cout<<m5.a<<m5.b<<"

";

hyong m6; m6=m; //把m的值赋给m6,注意这里不会调用复制构造函数,也不会生成临时对象,因为这里会把m当成是默赋值操作符的一个参数,调用的是默认赋值操作符。}

例:直接调用类中的构造函数

class A {public: int a; A(){a=0;} A(int i){a=i;} ~A(){cout<<"xi"<<"

";}

A f(){return A(3);} }; //在类中调用类的构造函数,当该函数被对象调用时反回由构造函数A构造的一个临时对象。

A g() {return A(5);} //类外的函数调用类的构造函数的方法,注意,这里是直接使用函数名的。

int main()

{ A m(1); A n(2); n=m.f(); cout<<n.a<<"

"; //输出3,调用类中的f函数,f函数用构造函数反回一个临时对象。

n=g(); cout<<n.a<<"

"; //输出5,调用类外的函数g

n=A(6); cout<<n.a; //在main函数中直接调用构造函数创造一个临时对象,然后把这个临时对象的值赋给对象n。

//n.A(7); } //错误,不能用类的对象来调用构造函数。

 
 
 
免责声明:本文为网络用户发布,其观点仅代表作者个人观点,与本站无关,本站仅提供信息存储服务。文中陈述内容未经本站证实,其真实性、完整性、及时性本站不作任何保证或承诺,请读者仅作参考,并请自行核实相关内容。
2023年上半年GDP全球前十五强
 百态   2023-10-24
美众议院议长启动对拜登的弹劾调查
 百态   2023-09-13
上海、济南、武汉等多地出现不明坠落物
 探索   2023-09-06
印度或要将国名改为“巴拉特”
 百态   2023-09-06
男子为女友送行,买票不登机被捕
 百态   2023-08-20
手机地震预警功能怎么开?
 干货   2023-08-06
女子4年卖2套房花700多万做美容:不但没变美脸,面部还出现变形
 百态   2023-08-04
住户一楼被水淹 还冲来8头猪
 百态   2023-07-31
女子体内爬出大量瓜子状活虫
 百态   2023-07-25
地球连续35年收到神秘规律性信号,网友:不要回答!
 探索   2023-07-21
全球镓价格本周大涨27%
 探索   2023-07-09
钱都流向了那些不缺钱的人,苦都留给了能吃苦的人
 探索   2023-07-02
倩女手游刀客魅者强控制(强混乱强眩晕强睡眠)和对应控制抗性的关系
 百态   2020-08-20
美国5月9日最新疫情:美国确诊人数突破131万
 百态   2020-05-09
荷兰政府宣布将集体辞职
 干货   2020-04-30
倩女幽魂手游师徒任务情义春秋猜成语答案逍遥观:鹏程万里
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案神机营:射石饮羽
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案昆仑山:拔刀相助
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案天工阁:鬼斧神工
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案丝路古道:单枪匹马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:与虎谋皮
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:李代桃僵
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案镇郊荒野:指鹿为马
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:小鸟依人
 干货   2019-11-12
倩女幽魂手游师徒任务情义春秋猜成语答案金陵:千金买邻
 干货   2019-11-12
 
推荐阅读
 
 
 
>>返回首頁<<
靜靜地坐在廢墟上,四周的荒凉一望無際,忽然覺得,淒涼也很美
© 2005- 王朝網路 版權所有