默认构造函数的构造操作

以下四种情况,编译器将会合成(即由编译器负责)一个nontrivial默认构造函数。

1、成员对象带有默认构造函数

例如:

#include <iostream>

using namespace std;

class Foo {
public:
    Foo() { cout << "Foo" << endl; }    // Foo类含有默认构造函数
};

class Bar {
public:
    Foo foo;
    int x;
};

int main()
{
    Bar bar;
    cout << bar.x;
    return 0;
}

运行结果:

带有默认构造函数的成员foo被初始化,内置类型x仍为垃圾值。注意,当x被显示初始化而foo未被显示初始化,编译器照样添加调用foo默认构造函数的代码对foo进行隐式初始化。

2、基类带有默认构造函数

如果一个类没有任何构造函数,但它的基类存在默认构造函数,那么编译器会合成nontrivial默认构造函数,调用基类的默认构造函数。

#include <iostream>

using namespace std;

class Base {
public:
    Base() { cout << "Base" << endl; }
};

class Foo {
public:
    Foo() { cout << "Foo" << endl; }    // Foo类含有默认构造函数
};

class Bar: public Base {
public:
    Foo foo;
    int x;
};

int main()
{
    Bar bar;
    cout << bar.x;
    return 0;
}

运行结果:

Bar存在一个Base基类和一个Foo成员,编译器合成一个默认构造函数,先调用基类的默认构造函数,再调用成员对象的默认构造函数,内置类型依旧没有初始化。

3、类中存在虚函数

当类中定义了虚函数,编译器会进行以下操作:

  • 建立一个virtual function table(vtbl),放虚函数的地址。
  • 在对象中插入一个vptr指向上面的vtbl。

这在另一篇文章“C++对象模型”中已经有所说明。这里需要注意的是,编译器会在类中的每一个构造函数类添加代码执行上述操作,如果没有定义构造函数,则编译器合成默认构造函数执行上述操作。

4、继承自虚基类的类

虚基类中的数据成员能够通过继承层次中的派生对象存取,但虚基类又只存在一个实例,所以各个派生类中必须要有能够记录虚基类地址的信息,这也是由编译器在构造函数中添加代码来完成的。特别的,当用户未定义任何构造函数,编译器会合成一个默认构造函数添加上述信息。

测试例程:

#include <iostream>

using namespace std;

class A {
public:
    int x;
};

class B: virtual public A {};

class C: virtual public A {};

class D: virtual public B, virtual public C {};

int main()
{
    D d;
    B b;
    C c;

    cout << &(b.x) << endl;
    cout << &(c.x) << endl;
    cout << &(d.x) << endl;
    cout << &(d.B::x) << endl;
    cout << &(d.C::x) << endl;

    return 0;
}

运行结果:

可以看到,不同对象中虚基类不是共享的,而同一继承层次中的虚基类是共享的。对象d、b、c都需要存储各自虚基类的地址。此任务交由编译器完成。

总结:

当类中没有显式定义构造函数,而又不是上述四种情况之一,那么编译器不会合成默认构造函数。所以说,未初始化的局部作用域的内置类型或复合类型都为垃圾值,对它们的初始化由程序员负责,而不是编译器。所以,当没有定义构造函数时,编译器会合成一个默认构造函数的说法是错误的。

环境:

Win7 + Code::Blocks

参考:

《深度探索C++对象模型》 P37-P47.

默认构造函数的构造操作

时间: 2024-09-29 04:03:01

默认构造函数的构造操作的相关文章

C++对象模型之默认构造函数的构造操作

一个类,如果没有任何的用户声明的的构造函数,那么会有一个默认的构造函数被隐式地声明出来.这个被隐式声明的构造函数,究竟什么时候被合成.被编译器合成的默认构造函数究竟执行怎么样的操作,编译器如何处理用户定义的构造函数,就是本文要探讨的问题. 1.默认构造函数何时被合成 如果一个类没有任何的用户声明的构造函数,那么在当编译器需要的时候,编译器会为类合成一个默认的构造函数,它只用于执行编译器所需要的操作.注意,默认的构造函数是在编译器需要的时候被合成出来,而不是程序需要的时候,如果程序需要,则默认的构

复制构造函数的构造操作

今天真机调试的时候莫名其妙遇到了这样的一个问题: This product type must be built using a provisioning profile, however no provisioning profile matching both the identity "iPhone Developer" and the bundle identifier..... 具体如下图所示: 十分蛋疼, 发现不管是从网上下的demo, 还是自己的过程.凡事真机测试的时候都

C++对象模型之复制构造函数的构造操作

复制构造函数用于根据一个已有的对象来构造一个新的对象. 1.构造函数何时被调用 有三种情况会以一个对象的内容作为另一个类的对象的初值构造一个对象,分别是: 1)对一个对象做显示的初始化操作时,如 class X { ... }; X x; X xx = x; // 或 X xx(x); 2)当对象被当作参数传递给某个函数时 3)当函数返回一个类的对象时 2.默认的成员复制初始化 如果class没有提供一个显式的复制构造函数,当class的对象以另一个对象作为初值进行构造时,其内部是以这样的方式完

C++ 合成默认构造函数的真相

对于C++默认构造函数,我曾经有两点误解: 类如果没有定义任何的构造函数,那么编译器(一定会!)将为类定义一个合成的默认构造函数. 合成默认构造函数会初始化类中所有的数据成员. 第一个误解来自于我学习C++的第一本书 <C++ Primer>,在书中392页:“只有当一个类没有定义构造函数时,编译器才会自动生成一个默认构造函数”. 实际上这句话也没有说错,它说明了默认构造函数定义的必要非充分条件,然而却给当时初学C++的我造成了一定的误解. 第二个误解依旧来自于Primer中的一句话:“合成的

c++值构造兼默认构造函数

friend ostream& operator<< (ostream&,String&); public: String(const char* str=NULL); //赋值构造兼默认构造函数(char) String(const String &other); //赋值构造函数(String) String& operator=(const String&other);       //operator= String operator+(

深度探索C++对象模型之第二章:构造函数语意学之Default constructor的构造操作

C++新手一般由两个常见的误解: 如果任何class没有定义默认构造函数(default constructor),编译器就会合成一个来. 编译器合成的的default constructor会显示的设定“class内每一个data member的默认值” 一.编译器在哪种情况下才会合成默认构造函数: 对于未声明构造函数的类,只有在以下四种情况下编译器才会为它们合成默认构造函数: 类的成员有一个类对象(Member Class Object),且该成员含有默认构造函数(default Const

C++:派生类的默认构造函数和拷贝构造函数调用基类构造函数的机制(含程序验证)

1.如果基类定义了不带参数的默认构造函数,则编译器为派生类自动生成的默认构造函数会调用基类的默认构造函数. 2.如果基类定义了拷贝构造函数,则编译器为派生类自动生成的拷贝构造函数同样会调用基类的拷贝构造函数. 3.如果基类定义了带参数的构造函数,派生类没有定义任何带参数的构造函数,则不能直接调用基类的带参构造函数,程序编译不通过. 代码如下: #include<iostream> using namespace std; //基类Game,定义了两个构造函数和一个拷贝构造函数 class Ga

【C/C++】构造函数、默认构造函数、成员初始化列表

常见问题 Q1. 下列关于构造函数的描述中,错误的是( ) A. 构造函数可以设置默认的参数 B. 构造函数在定义类对象时自动执行 C. 构造函数可以是内联函数 D. 构造函数不可以重载 Q2. 下列代码中a.b的各个成员变量值是多少? 1 class Student 2 { 3 public: 4 Student() {} 5 void show(); 6 private: 7 string name; 8 int number; 9 int score; 10 }; 11 Student a

C++基础知识—关于默认构造函数的一切

C++条款-关于默认构造函数的一切 默认构造函数(defaultconstructor): 默认构造函数是在未提供显示初始值时,用来创建对象的构造函数.如果没有提供任何构造函数,则C++将自动提供默认构造函数,默认构造函数没有参数,因为声明中不包含值.同时,当且仅当没有定义构造函数时,编译器才会提供默认构造函数.为类定义了构造函数后,程序员就必须为它提供默认构造函数,如果提供了非默认构造函数,但没有提供默认构造函数,则在初始化对象时调用默认构造函数是会出错的: Stack   ww;// 错误