论坛登陆 用户: 密码:
联系我们
设为首页
加入收藏
业界新闻 网络编程 程序开发 网页图象 聊天通讯 软件应用 网络安全 硬件学堂 教育频道 站长club
  ·推荐新闻
 
·美前任官员认为amd告倒英特
·搜索引擎关键字排行简介
·网站优化教程(一)
·msn近期遭受木马病毒骚扰 用
·用qq管理你的系统^_^ 
·测评中心金山毒霸联合发布7
·qq群聊实名 普通用户不受影
·雅虎思科联手推数字邮件签名
·这18条背下来没人敢和你忽悠
·自己动手,拯救丢失的硬盘数
  ·资料搜索
 
  ·相关文章
·深度探索c++对象模型(完结篇)
·深度探索c++对象模型(10)
·深度探索c++对象模型(9)
·深度探索c++对象模型(8)
·深度探索c++对象模型(7)
·深度探索c++对象模型(6)
·深度探索c++对象模型(4)
·深度探索c++对象模型(3)
·深度探索c++对象模型(2)
  ·热门新闻
首页>>程序开发>>c/c++区>>文章正文

深度探索c++对象模型(5)


 日期:2005-7-18 9:14:08     来源:易特网络技术   编辑:黑鹰  点击:
上一篇我们对合成确省的构造函数做了一个了解,这一篇我们继续看看构造函数这个有趣的东西.
copy constructor是什么?我们经常看到代码中有一些这样的函数调用方式x(x&) (“x of x ref”). 这个函数用用户自定义类型作为参数,那它的参数的构造便是由copy constructor负责的. 可见这个玩意非常重要,实际上copy constructor是由编译器自动合成的,不需要你去作任何事情,但编译器都做了些什么呢?我们的问题出来了。

  我们有三种情况需要用一个对象的内容作为另一个类对象的初值.也就是需要编译器来为我们自动合成copy constructor.一种是我们在编程中肯定回用到的由类生成对象例如以下形式:

class classa{......}
classa a;
classa b=a; //一个class对象以另一个对象做初值

  另外的一种情况是以对象为参数在函数中传递看下面的伪码:

//例如我们有一个cuser类
cuser{
cuser();
......
};
//我们还有一个cdatabase类,它有一个addnew的方法
cdatabase{
......
public:
addnew(cuser userone);
......}
//我们用cuser类产生了一个对象实例.userone,并将他作为addnew函数的参数,以便
//addnew函数能够完成在数据库中增加一条记录,用来记录一个用户的信息
cdatabase db=new cdatabase();
db.addnew(cuser userone) //在这里,你不用将你的用户类的成员全部展开.

  还有一种当然是用做函数的return,例如你可以在cdatabase类中添加一个函数用来读取一个用户的信息例如这样cuser getuserone(int userid),通过一个用户的唯一的编号可以获得一个用户的信息,并返回一个cuser类的对象。

  我们来看看copy constructor是如何工作的.首先copy constructor和default constructor一样都是在需要的时候由编译器产生出来,一个类如果没有声明一个copy constructor就会存在一个隐含的声明(或定义),它也被分为trivial和nontrivial两种.

  我们来看书上的例子:

class word
{
public:
word(const char*);
~word(){delete [] str;}
private:
int cnt;
char *str;
}
这个类的声明不需要合成出default copy constructor.但当进行如下应用时:
#include "word.h"
word noun("lsmodel");
void foo()
{
word verb=noun;
}

  结果将会出现灾难性的后果.为什么?因为我们的逻辑对象verb和全局对象noun都指向了相同的字符串,在退出函数foo()之前verb会执行析构,则字符串被删除,从此全局对象nonu指向了一堆无意义的东西.你可以声明一个explicit copy constructor来解决这个问题,当然还可以让编译器来自动的给你合成一个copy construct.

  我们将上面的word类改写成下面的样子:

class word
{
public:
word(const string&);//注意这里和我们开始的x(x&)形式一样
~word();
//......
private:
int cnt;
string str; // 这个成员是string类的对象,string是我们自定义的类型
};
class string
{
public:
string(const char*);
string(const string&);//这里声明了一个copy constructir
~string();
//......
}
这时在执行我们的代码
#include "word.h"
word noun("lsmodel");
void foo()
{
word verb=noun;
}

  编译器会为我们的word类合成一个copy constructor,用来调用它的str(member class string object)的copy constructor.象下面伪码表示的这样:

inline word::word(const word &wd)
{
str.string::string(wd.str);
cnt=wd.cnt;
}

  当这个类中有一个或多个虚函数时,或者这个类是派生于一个继承串链,并且这个串中有一个或多个虚拟的基类时.这个类在进行拷贝时便不会展现逐次拷贝(bitwise copy).并且会通过合成的copy constructor来重新明确的设定vptr来指向虚函数表,而不是将右边对象的vprt直接拷贝过来.书上的zooanimal例子的图可以很清晰的描述出这点。

  如果一个对象以另一个对象做初值,而后者有一个virtual base class subobject,那会怎样呢?任何一个编译器都会做到在派生类对象中的virtual base class subobject的位置在执行期就准备妥当,但bitwise copy可能会破坏这一位置,因此也需要由编译器合成出一个copy constructor,来安插一些代码来设定virtual base class pointer/offset,对每一个成员执行必要的memberwise初始化操作,以及执行内存相关的工作。

  最后我们来总结一下上面说的内容,确实有些乱.雷神越来越觉得自己的缺乏文字描述能力.

  我们这篇学习的内容是:当一个对象以另一个对象作为初始值时,会发生什么事情.

  分成了两种情况,一种是我们声明了explicit copy constructor,这个不是这篇文章需要搞明白的(我想大家也都很明白了).我们想知道的是我们没有为class声明explicit copy constructor函数时编译器都干了些什么.编译器会为我们合成一个copy constructor.以便适应任何时候的对象被正确的初始化.并且我们了解了有以下四种情况class不在按位逐一进行拷贝.

  1.当你设计的类声明了一个explicit copy constructor函数时.
  2.当你设计的类是由一个具有explicit copy constructor的基类派生的时.
  3.当你设计的类声明了一个或多个虚函数时.
  4.当你设计的类派生自一个继承串链,这个继承串链中有一个或多个virtual base classes时.

  好了,就到这里吧,休息,休息一下。

上一篇:深度探索c++对象模型(4)
下一篇:深度探索c++对象模型(6)
[发送给好友] [打印本页] [关闭窗口] [返回顶部转载请注明来源:http://www.iyit.net
特别声明: 本站除部分特别声明禁止转载的专稿外的其他文章可以自由转载,但请务必注明出处和原始作者。文章版权归文章原始作者所有。对于被本站转载文章的个人和网站,我们表示深深的谢意。如果本站转载的文章有版权问题请联系编辑人员,我们尽快予以更正。
责任编辑: 黑鹰 投稿作者: 易特网络
信息来源: 易特网络技术 录入时间: 2005-7-18 9:14:08
浏览次数: 投稿信箱: shtghy@163.com
设置首页 - 版权声明 - 广告服务 - 关于我们 - 联系我们 - 友情连接
copyrights ©2004-2005 iyit.net all rights reserved. 网站合作、广告联系qq:147007642、466949678
易特网络技术 点击这里给我发消息