C++拷贝构造函数语意学 copy constructor(The Semantics of Constructors)
2015-10-26 20:27
513 查看
使用copy constructor的三种情况
有三种情况,会以一个object的内容作为另一个class object的初值。1.显式的以一个object的内容作为另一个class object的初值
class X{......}; X x; X xx = x;
2.当object被当做参数交给某个函数时
void foo(X x); void bar(){ X xx; foo(xx); }
3.当函数传回一个class object时
X foo(){ X xx; return xx; }
Explicit Copy Constructor
用户显式定义的copy consyructor大部分情况下以另一个同类实例作为初值。如:
X::X(const X& x); Y::Y(const Y& y,int = 0);
并注意传入参数必须是reference的
因为如果定义如下copy consyructor
X::X(X x)//ERROR: Copy constructor must pass its first argument by reference
X(X x)是传值函数。把形参复制到实参时需要调用copy constructor ,会造成无休止的递归。
Default Memberwise Initialization(默认逐位初始化)
如果class没有提供一个Explicit Copy Constructor时,当class object 以相同的class的另一个object作为初值,其内部是以Default Memberwise Initialization手法完成的,也就是把每一个内建的或派生的data member的值,从某个object拷贝一份到另一个object身上。不过并不会拷贝member class object,而是以递归的方式施行memberwise initialization.例如:
class MyString{ public: //没有Explicit Copy Constructor private: char* str; int len; };
发生如下操作时
MyString noun("book"); MyString verb = noun;
其方式就像如下:
verb.str = noun.str; verb.len = noun.len; //member class object以递归的方式施行memberwise initialization
No Bitwise Copy Semantics(不位逐次拷贝)
若一个class不展现出bitwise copy semantics时,编译器需合成copy constructor,而非简单的 bitwise copy。下面四种情况将展现 No Bitwise Copy Semantics.1.当class内含一个声明有 copy constructor(不论是显式声明还是由编译器合成)的member object时。
如:
class X{ public: X(X& x){ printf("X copy constructor\n"); } }; class Y{ public: X xx; };
Y被合成出的copy constructor类似这样:
inline Y::Y(const Y& yy){ this.xx.X::X( yy.xx );//调用X的copy constructor传入yy的xx成员。将结果赋予this.xx }
2.当class继承自一个存在 copy constructor 的 base class
详情同上
3.当class声明了一个或多个virtual functions时
当class声明了一个或多个virtual functions时,编译器需对每一个新产生的class object 的 vptr 进行正确的设置。
我们定义两个class如下:
class ZooAnimal{ public: ZooAnimal() = default; virtual void animate(); virtual void draw(); }; class Bera: public ZooAnimal{ public: void animatr(); void draw(); virtual void dance(); };
ZooAnimal class object 以另一个ZooAnimal class object 作为初值,或Bear class object 以另一个Bear class object 作为初值,都可直接靠bitwise copy semantics 完成,因为其vprt指向同一位置。
如:
Bear yogi; Bear winnie = yogi;//可直接靠bitwise copy semantics 完成
winnie的vprt应指向Bear class的virtual table。而yogi也是如此,所有使用逐位拷贝是安全的。
编译器也需保证一个base class object以其derive class 的 object 内容做初始化操作时vptr的复制是安全的。
Bear yogi; ZooAnimal frannny = yogi;//此时vptr的操作应是安全的
合成出的ZooAnimal copy constructor会设定object的vptr,以指向ZooAnimal的virtual table。
4.当class派生自一个继承串联,其中有一个或多个virtual base class时
编译器需在copy constructor调用virtual base class 的default constructor ,并设置vptr 和 定位派生类中 virtual base class subobject。
相关文章推荐
- 使用C++实现JNI接口需要注意的事项
- 关于指针的一些事情
- c++ primer 第五版 笔记前言
- share_ptr的几个注意点
- Lua中调用C++函数示例
- Lua教程(一):在C++中嵌入Lua脚本
- Lua教程(二):C++和Lua相互传递数据示例
- C++联合体转换成C#结构的实现方法
- C++编写简单的打靶游戏
- C++ 自定义控件的移植问题
- C++变位词问题分析
- C/C++数据对齐详细解析
- C++基于栈实现铁轨问题
- C++中引用的使用总结
- 使用Lua来扩展C++程序的方法
- C++中调用Lua函数实例
- Lua和C++的通信流程代码实例
- C与C++之间相互调用实例方法讲解
- 解析C++中派生的概念以及派生类成员的访问属性
- C++ Custom Control控件向父窗体发送对应的消息