首页 > 代码库 > C++基础:缺省构造函数

C++基础:缺省构造函数

        缺省构造函数,又称默认构造函数,是C++以及其他的一些面向对象的程序设计语言中,对象的不需要参数即可调用的构造函数。下面将针对缺省构造函数的定义、使用、以及注意问题等方面简要探讨。

1、缺省构造函数是怎样的形式?是如何定义的?

         在C++的一个类中,如果构造函数没有参数,或者构造函数的所有参数都有默认值,就可以称其为缺省构造函数。一个类中,只能有一个缺省构造函数

  • 对于如下代码:构造函数MyClass()没有参数,就是MyClass的缺省(默认)构造函数。
class MyClass
{
public:
    MyClass(){
        value = 1;
        cout<<"无参默认构造函数调用并初始化value:"<<value<<endl;
    };
private:
    int value;
};
  • 而对于如下代码:构造函数Myclass(int  i = 10)的所有参数(此处只有一个参数i)都有默认值,这样的形式,也是MyClass的缺省(默认)构造函数。
class MyClass
{
public:
    MyClass(int i = 10){
        value = i;
        cout<<"参数都有默认值的默认构造函数调用并初始化value:"<<value<<endl;
    };
private:
    int value;
};
  • 要注意,一个类中只能有一个缺省构造函数。如果将以上的两个构造函数均加入类中,形如以下代码,就会有问题。
class MyClass
{
public:
    MyClass(){
    value = 1;
    cout<<"无参默认构造函数调用并初始化value:"<<value<<endl;
    };
    MyClass(int i = 10){
        value = i;
        cout<<"参数都有默认值的默认构造函数调用并初始化value:"<<value<<endl;
    };
private:
    int value;
};


       此时当建立MyClass mc这样的对象的时候,就会报错:

2、缺省构造函数在何时被调用?

缺省构造函数的调用主要包括如下的几种情况:

  • 对象被定义时无参数,形如:MyClass mc;
  • 动态分配对象时无参数列表,形如:MyClass *mcptr = new MyClass; 或 MyClass *mcptr = new MyClass();
  • 使用静态分配的数组,数组元素为某个类的对象,形如: MyClass mcArrayStatic[5];
  • 使用动态分配的数组,数组元素为某个类的对象,形如:MyClass *mcArrayDynamic = new MyClass[6];
  • 使用标准库的容器(如vector),容器的元素类型是某个类的对象,形如:vector<MyClass> vmc(7);
  • 在派生类的构造函数中未显示调用基类的构造函数,此时基类的缺省构造函数会被调用

        下面通过一段代码加以演示:

#include<iostream>
#include<vector>
using namespace std;

class MyClass
{
public:
    MyClass(){
    value = 1;
    cout<<"无参默认构造函数调用并初始化value:"<<value<<endl;
    };
    /*MyClass(int i = 10){
    value = http://www.mamicode.com/i;"参数都有默认值的默认构造函数调用并初始化value:"<<value<<endl;
    };*/
private:
    int value;
};

class MyClassChild:public MyClass
{
public:
    MyClassChild(){
        cout<<"派生类构造函数调用"<<endl;
    }
};

class MyClass2
{
public:
    MyClass2(){
        cout<<"Myclass2构造函数调用"<<endl;
    }
private:
    MyClass myc;
};

//用于对缺省构造函数调用情况的演示测试,在main()函数直接调用即可
void test_myclass()
{
    cout<<"对于默认构造函数调用的测试:"<<endl;
    cout<<"\n未显示调用构造函数:"<<endl;
    MyClass mc;
    cout<<"\n显示调用构造函数:"<<endl;
    MyClass *mcptr = new MyClass;
    //或MyClass *mcptr = new MyClass();
    cout<<"\n静态分配数组:"<<endl;
    MyClass mcArrayStatic[5];
    cout<<"\n动态分配数组:"<<endl;
    MyClass *mcArrayDynamic = new MyClass[6];
    cout<<"\n使用标准库容器:"<<endl;
    vector<MyClass> vmc(7);//最大容量为7
    cout<<"\n创建派生类:"<<endl;
    MyClassChild mcc;
    cout<<"\n包含别的类成员的类的创建:"<<endl;
    MyClass2 mc2;
}

         在main()函数调用test_myclass(),得到如下运行结果:

    3、注意事项:

        (1)对于你所创建的类(如此处的MyClass),如果没有为该类提供任何构造函数,那么编译器会自动生成一个默认的无参数的构造函数,但该函数体为空,不进行任何工作;然而,只要你为你的类定义了构造函数,无论是默认的构造函数形式,还是非默认构造函数的形式,哪怕只有一个,编译器都不会再生成默认的构造函数。

        举例言之,对于以上的类MyClass(),我去掉所有的构造函数,在运行下面的test_myclass(),依然可以运行,这是因为此时编译器为我的类生成了默认的构造函数,只是在其中未进行任何操作:

class MyClass
{

private:
    int value;
};

        (2)在需要调用默认构造函数的情况下,如果并不存在默认构造函数,就会出错。

        比如,我定义了非默认构造函数形式的构造函数,此时编译器并不会帮我生成默认的构造函数,若运行test_myclass(),在需要调用默认构造函数的情况下,就会出错。

class MyClass
{
public:
    /*MyClass(){
    value = http://www.mamicode.com/1;"无参默认构造函数调用并初始化value:"<<value<<endl;
    };*/
    MyClass(int i ){//没有默认值,就不是默认构造函数
    value =http://www.mamicode.com/ i;
    cout<<"参数都有默认值的默认构造函数调用并初始化value:"<<value<<endl;
    };
private:
    int value;
};


        (3)并非所有的类都存在默认构造函数,前面已经说过,如果现实的定义了非默认构造函数,则编译器就不会自动生成默认构造函数,也就不存在默认构造函数,这是常见的错误,要注意。

        (4)如果某派生类的基类没有缺省构造函数,那么编译器也不会为该类隐式地定义缺省构造函数。因为该类即使隐式地定义缺省构造函数,也无法初始化其基类。

        (5)如果类的基类的缺省构造函数为私有,那么编译器会为该类隐式地定义缺省构造函数,但编译报错“cannot access private member declared in class 基类名字”。