C++关于编译器合成的默认构造函数
有两个常见的误解:
1.任何类如果没有定义默认构造函数,就会被合成出一个来。
2.编译器合成的默认构造函数会显式地设定类内每一个数据成员的默认值。
对于第一个误解,并不是任何类在没有显式定义默认构造函数时都会被编译器合成一个默认构造函数。
在以下4种情况下,编译器才会合成默认构造函数,以满足编译器自己的需求(并不是为了满足程序的需求)。
一、父类有默认构造函数(default constructor)
如果一个没有任何构造函数的类派生自 “一个有默认构造函数的” 父类,那么这个派生类的默认构造函数被认为是 ”有用的(被编译器所需求)“,因此需要被合成出来。它会调用父类的默认构造函数。
如果派生类含有多个构造函数,但其中不含默认构造函数,编译器并不会为他合成新的默认构造函数,而是会扩展每一个构造函数,将所有需要调用的默认构造函数的代码安插进去。
二、类中带有类类型成员
如果类A中带有类类型成员,并且这个类类型是有默认构造函数的,那么类A的默认构造函数被认为是 ”有用的“,需要合成。
//X,Y,Z都带有显式默认构造函数
class X
{
public:
X();
};
class Y
{
public:
Y();
};
class Z
{
public:
Z();
};
class A
{
public:
X x; //三个类类型
Y y;
Z z;
A(int a); //带有一个构造函数
private:
int val;
}; //程序员对A的构造函数的实现(你所看到的)
A::A(int a)
{
val = a;
} //编译器扩展合成后(编译器认为应该这样)
A::A(int a)
{
//按声明顺序安插代码,调用构造
x.X::X();
y.Y::Y();
z.Z::Z(); //显式的用户代码
val = a;
}
三、带有虚函数的类
在编译期间会发生两个扩展行动:
1.编译器会产生一个虚表(存放着类内虚函数的地址)。
2.在每一个类对象中,会有一个额外的虚表指针被编译器合成出来,用来指向相关虚表。
四、带有虚基类的类
虚基类的实现必须满足虚基类在其 ”每一个派生子类的对象中的位置“ 能够于执行期准备妥当。
class A
{
public:
int a;
};
class X :public virtual A
{
public:
int x;
};
class Y :public virtual A
{
public:
int y;
};
class Z :public X, public Y
{
public:
int z;
}; //编译时无法确定p->A::a的位置
void fun(X *p){ p->a = ; } //编译时无法确定p的真正类型(基类指针可以指向派生类对象,此时真正类型为派生类类型) int main()
{
fun(new X); //真正类型为X
fun(new Z); //真正类型为Z
return ;
}
编译器无法确定fun()中 ”经由p存取的A::a “的实际偏移位置,因为p的真正类型是可变的,如在main()中既可以是X类型,也可以是Z类型。编译器必须改变存取操作的代码,使A::a延迟到执行时才确定下来。
fun()可以被编译器改写成如下:
//编译器转变操作,其中vcbA是编译器产生的指针,指向虚基类A
void fun(X *p){ p->vcbA->a = ; } //vcbA是在类对象构造期间被完成的
除此四种情况之外,如果类没有声明任何构造函数,他们就会有一个隐式而”无用“的默认构造函数,他们实际上并不会被合成构造出来。
对于第二个误解,在合成的默认构造函数中,只有基类子对象和类类型对象会被初始化,而其他所有的非静态成员(如整数,指针,数组等),都不会初始化,对他们进行初始化的应该是程序员,而非编译器。
注意:值类型的默认值并不是默认构造的初始化。
C++关于编译器合成的默认构造函数的更多相关文章
- 合成的默认构造函数定义为delete的一种情况(针对C++11标准)
1. 默认初始化 如果定义变量时没有指定初值,则变量会被默认初始化,此时变量被赋予了"默认值". 对于类类型的变量来说,初始化都是依靠构造函数来完成的.因此,即使定义某个类的变量( ...
- c++ 2.1 编译器何时创建默认构造函数
我们通常会说当生命一个 class 时,如果我们不为该 class 指定一个 constructor,那么编译器会替我们实现一个 connstructor,那么这种说法一定对吗? 事实上,这是不对的. ...
- C++ 合成默认构造函数的真相
对于C++默认构造函数,我曾经有两点误解: 类如果没有定义任何的构造函数,那么编译器(一定会!)将为类定义一个合成的默认构造函数. 合成默认构造函数会初始化类中所有的数据成员. 第一个误解来自于我学习 ...
- C++编译器合成Default Constructor的4种情况
笔记C++编译器为编译器需要合成Default Constructor的4种情况. 1,Class A内含Class B对象,Class A没有Default Constructor时会在编译时合成D ...
- C++对象模型——默认构造函数的合成
最近在学习C++对象模型,看的书是侯捷老师的<深度探索C++对象模型>,发现自己以前对构造函数存在很多误解,作此笔记记录. 默认构造函数的误解 1.当程序猿定义了默认构造函数,编译器就会直 ...
- C++中默认构造函数中数据成员的初始化
构造函数的任务是初始化数据成员的,在类中,如果没有显示定义任何构造函数,编译器将为我们创建一个构造函数,称为合成的默认构造函数,合成的默认构造函数使用与变量初始化相同的规则来初始化成员.即当类中的数据 ...
- 【C++对象模型】构造函数语意学之一 默认构造函数
默认构造函数,如果程序员没有为类定义构造函数,那么编译器会在[需要的时候]为类合成一个构造函数,而[需要的时候]分为程序员需要的时候和编译器需要的时候,程序员需要的时候应该由程序员来做工作,编译器需要 ...
- C++对象模型的那些事儿之三:默认构造函数
前言 继前两篇总结了C++对象模型及其内存布局后,我们继续来探索一下C++对象的默认构造函数.对于C++的初学者来说,有如下两个误解: 任何class如果没有定义default constructor ...
- [C++11] 默认构造函数
类通过一个特殊的构造函数来控制默认初始化过程.这个函数就是默认构造函数.默认构造函数无需不论什么实參. 我们能够显示的定义默认构造函数也能够让编译器为我们生成默认构造函数. 默认构造函数以例如以下规则 ...
随机推荐
- 算法 python实现(三) 快速排序
算法学起来真费劲啊,智商只够捉只鸡的.昨晚没看明白就没电了,过两天要考虑偷电了... 今天看看快速排序,有一个博客写的很好,通俗生动形象,适合我这样的算法大白菜.推荐一下 http://www.cnb ...
- 理解C++11正则表达式(2)
今天有幸(2016/3/19)在上海参加了C++交流会,见到了梦寐已久想见的台湾C++大神老师侯捷,心情十分的激动.侯老师对C++理解的深刻,让人叹为观止.以为他教学的严谨,说话方式娓娓道来,听着非常 ...
- UILable自适应frame
UILabel *textlab = [[UILabel alloc]initWithFrame:CGRectMake(20, 10,ScrollView.frame.size.width - 40, ...
- hdoj 3952 World Exhibition
World Exhibition Time Limit: 2000/1000 MS (Java/Others) Memory Limit: 32768/32768 K (Java/Others) ...
- MediaInfo源代码分析 2:API函数
本文主要分析MediaInfo的API函数.它的API函数位于MediaInfo.h文件中的一个叫做MediaInfo的类中. 该类如下所示,部分重要的方法已经加上了注释: //MediaInfo类 ...
- spfa + slf优化
最近在练习费用流 , 不是要用spfa吗 ,我们教练说:ns学生写朴素的spfa说出去都让人笑 . QwQ,所以就去学了一下优化 . slf优化就是双向队列优化一下,本来想用lll优化,可是优化后我t ...
- use SWF / Flash in cocos2d-x; cocos2d(cocos2d-x) 直接播放flash / SWF文件
前段时间移植一个页游到手游,原先页游的项目里面有1000+的Flash人物,宠物动画,特效. 这要是全部重新做一遍,还不累死人?所以就想干脆直接在Cocos2d(x)里面播放SWF文件.(包括场景,过 ...
- WinForm TextBox自定义扩展方法数据验证
本文转载:http://www.cnblogs.com/gis-crazy/archive/2013/03/17/2964132.html 查看公司项目代码时,存在这样一个问题:winform界面上有 ...
- 谷歌google搜索打不开、谷歌gmail邮箱及相关服务无法登录的解决的方法
歌打不开 google打不开,与中国大陆封杀有关,可是主要是由于近期googleserver在全球范围内又一次进行了布局调整. 解决的方法是仅仅要改动用户本地计算机hosts文件就能够了. 一.Win ...
- 应用Druid监控SQL语句的执行情况--转载
Druid是什么? Druid首先是一个数据库连接池.Druid是目前最好的数据库连接池,在功能.性能.扩展性方面,都超过其他数据库连接池,包括DBCP.C3P0.BoneCP.Proxool.JBo ...