您的位置:首页 > 编程语言 > C语言/C++

c++变量初始化(猜想......)

2009-07-24 16:58 204 查看
全局的、名字空间的、静态的对象会被自动初始化适当类型的0。

堆,容器中和类的显式空构造函数的对象会被调用其构造函数初始化,但并不会被初始化为相应的0,(没有数组)如

char ch[10];

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

cout<<(int)ch[i]<<endl;//-52 -52 -52....int数组亦然

结构和类成员的初始化:

谈到结构和类成员,除非提供显示的构造函数进行初始化,否则可以分为以下两种(是否用默认构造函数初始化):

class A

{

//A(){}

public:

int i;

}

A a;

void f()

{

A a1;

cout<<a.i;//error,因为A并没有提供nontrivial default constructor,所有并没有初始化

cout<<a1.i;//0, 因为a是全局变量。

}

如果将//A(){}去掉,则cout<<a.i;会打印出一个未定义的i值,并不会报错。

总之,用户如果没有提供一个构造函数的话,但只是为类中的类成员和基类初始化!!!其他的一概不初始化!!!

附录:

nontrivial default constructor信息参见以下:

http://blog.csdn.net/dongfengsun/archive/2008/02/18/2104176.aspx

假期间闲来无事,就下载了某大师的VC++视频资料。在讲到C++时,说是如果程序员没有自己定义默认构造函数,

那么编译器会自动为我们产生一个默认的构造函数。 本来这个错误的认识很多程序员都有,不足为奇。但有这

么多年编程经验的高手也有这样的错误认识就不禁让我哑然了。

其实编程语言和我们所用的任何软件没有区别,例如Photoshop、AutoCAD之类。其唯一不同的是我们用

的编程语言是基于编译器的,而应用软件是基于我们的编程语言的。

既然我们所用的软件是基于编译器的,那么理解编译器在背后到底为我们做了些什么、在什么情况下做

了哪些事情就显得异常重要。这就像Photoshop会为你产生一些基本图形例如矩形、三角形之类,而不会凭空产

生一些风景优美的图片一样。

在《C++ Annotated Reference Manual(ARM)[ELLIS90]》中的Section 12.1告诉我们:"Default

constructors...在需要的时候被编译器产生出来"。

其实默认构造函数也是分为两类的:有用的、无用的。

所谓有用的标准也是就默认构造函数会为我们的类做一些初始化操作。那么无用的就不会做任何工作,

从而对我们的类也就没有任何意义。所以,我们通常所说的默认构造函数是指有用的默认构造函数,其英文名字

叫nontrivial default constructor。

那么到底什么时候编译器会为我们产生nontrivial default constructor呢?有下面四中情况:

①如果一个类里面某个成员对象有nontrivial default constructor,编译器就会为我们的类产生

nontrivial default constructor。

那么编译器这样做的理由是什么?

答案是因为类成员对象有nontrivial default constructor,那么编译器就需要显式的来调用这个类成

员对象的nontrivial default constructor。而编译器想显式的调用类成员对象的nontrivial default

constructor,就需要自己来合成一些代码来调用。但是记住,编译器合成的nontrivial default constructor仅

仅调用类成员对象的默认构造函数,而不对我们类里面的其它变量做任何初始化操作。

也就是说,如果你想初始化类成员变量以外的变量例如一个int、一个String,那么必须自己定义默认构

造函数来完成这些变量的初始化。而编译器会对你定义的默认构造函数做相应的扩展,从而调用类成员对象的

nontrivial default constructor。

②如果一个派生类的基类有nontrivial default constructor,那么编译器会为派生类合成一个

nontrivial default constructor。

编译器这样的理由是:因为派生类被合成时需要显式调用基类的默认构造函数。

③如何一个类里面隐式的含有任何virtual function table(或vtbl)、pointer member(或vptr)。

编译器这样做的理由很简单:因为这些vtbl或vptr需要编译器隐式(implicit)的合成出来,那么编译器

就把合成动作放到了默认构造函数里面。所以编译器必须自己产生一个默认构造函数来完成这些操作。

所以如果你的类里带有任何virtual function,那么编译器会为你合成一个默认构造函数。

④如果一个类虚继承于其它类。

编译器这样做的理由和③类似:因为虚继承需要维护一个类似指针一样,可以动态的决定内存地址的东

西(不同编译器对虚继承的实现不仅相同)。

那么除了以上四种情况,编译器并不会为我们的类产生默认构造函数。

所以编程中切忌想当然,要明白哪些事情是编译器做的,哪些事情需要程序员来完成的。就像堆所占用

的资源需要程序员自己来释放,而栈空间是编译器管理的一样。

只有如此,才能编写出质量更高的代码。
内容来自用户分享和网络整理,不保证内容的准确性,如有侵权内容,可联系管理员处理 点击这里给我发消息
标签: