一、认识构造函数
当创建一个类的对象时,主要是通过一个或者几个特殊的成员函数来控制对象的初始化,这种函数就是构造函数。它的任务就是用来初始化类的成员的,所以当创建类对象时候就会调用构造函数。
构造函数的几个特点:
- 函数名和类名必须一样,且没有返回值。
- 当没有显式的定义类的构造函数时,系统会自己生成默认的构造函数。
- 构造函数可以重载。
class Date
{
public:
Date()
{ }
Date(int day)
{
_year = 1949;
_month = 10;
_day = day;
}
void print()
{
cout << _year << "-" << _month << "-" << _day << endl;
}
private:
int _year=1990;
int _month;
int _day;
};
上面的代码中,定义了一个简单的Date类类型,可以看到有显式的给出了构造函数,第一个是没有参数列表且函数体不做任何操作的,还有一个是仅有一个整形参数day的,就是当我传入一个整形的day参数,则在函数内部就把它的_year和_month初始化为1949跟10,而_day初始化为传入的实参值。这样的两个构造函数就构成了重载,因为能够重载,所以在写构造函数的时候只能有一个参数列表为空的构造函数()
当不传参数的定义一个Date类型对象,会调用显式定义的缺省的构造函数,在没有初始化列表的情况下采取类内初始化或默认初始化,上面的程序中,如果不传入参数,那么构建的对象的_year成员初始化为1990,另外两个值随机。
我们在没有显式的定义构造函数时候,系统会自动生成一个默认构造函数。当我们定义了一些其他的构造函数时,这个类就将没有默认构造函数。所以当我们显式的定义了其他的构造函数,最好把默认构造函数也显式的定义一遍。
当我们定义的默认构函数,并不需要做什么事情的时候,只是因为需要满足上面所提的情况才显式的定义它时,那么此时的默认构造函数等同于系统生成的默认构造函数,那么我们可以这么写:
Date() = default;
因为在新标准中,如果需要系统默认的行为,就可以通过在参数列表后加上=default
来使编译器生成系统默认的构造函数。
二、初始化列表
如下面代码所示,在冒号:
跟花括号{}
之间的代码成为构造函数的初始值列表,它的作用是给创建的对象的某些成员变量赋初值。这种是在构建对象的时候初始化,是在对象创建成功之前完成的,和在函数体内赋值是不一样的,函数体内赋值是你的对象成员都已经创建好之后,再对成员进行赋值。
Date(int year=1990,int month=1,int day=1)
:_year(year), _month(month), _day(day), t(10)
{ }
可以看到,这种初始化并不是必须的,但是在以下几种情况时是必须的:
- 1.成员是const类型。
- 成员是引用类型。
- 有一个成员是类类型的对象(且它没有缺省的构造函数)。
class Time
{
public:
Time( )
{
}
private:
int _hour;
};
class Date
{
public:
Date(int year=1990,int month=1,int day=1)
:_year(year), _month(month), _day(day), t(10)
{ }
void print()
{
cout << _year << "-" << _month << "-" << _day << endl;
}
private:
int _year=1990;
int _month;
int _day;
Time t;
};
解释:
1.对于const和引用类型,必须要进行初始化,所以他们必须在初始化列表中进行初始化。
2.当类类型成员有缺省的构造函数时,在创建对象的时候体统会默认调用,因为不用传参。当你的构造函数不是缺省的,如果不在初始化列表中进行调用构造函数,系统就无法知道怎么调用t的构造函数,那么就无法创建t了。
如上代码中,需要在参数列表中调用t的构造函数才不会出错。
成员初始化顺序:
在上面的初始列表中,每个成员只能出现一次,因为一个变量多次初始化是无意义的。
还有重要的一点,初始化列表的顺序并不限定初始化的执行顺序。成员的初始化顺序是与类中定义的顺序保持一致。可以看看下面的初始化列表:
public:
Date(int year=1990,int month=1,int day=1)
:_year(year), _month(month), _day(day), t(10)
{ }
在这里的意思是想要用1来初始化_month,再用_month初始化_year。但其实是_year被先初始化,而此时你的_month并没有初始化,所以,最后的结果是_year是一个随机值。
所以,最好让构造函数初始值的顺序与成员声明的顺序保持一致。
网友评论