一文搞懂C++构造函数的初始化列表

简介: 我们在实例化对象后,对象自动调用构造函数来初始化自己的值,完成初始化工作,但是我们并不能称它为成员变量的初始化,因为初始化只能初始化一次,然而在函数体的“初始化”工作,准确的说可以赋值多次,所以把函数体赋值当作初始化,有失偏颇。因此引入初始化列表
class Time
{
public:
  Time(int hour, int min, int sec)
  {
    _hour = hour;
    _min = min;
    _sec = sec;
  }
private:
  int _hour;
  int _min;
  int _sec;
};

我们在实例化对象后,对象自动调用构造函数来初始化自己的值,完成初始化工作,但是我们并不能称它为成员变量的初始化因为初始化只能初始化一次,然而在函数体的“初始化”工作,准确的说可以赋值多次,所以把函数体赋值当作初始化,有失偏颇。

因此引入初始化列表

1 初始化列表

语法:以一个冒号开始,接着用逗号分隔数据成员列表,每个“成员变量”后面跟一个放在括号中的初始值或表达式。

具体格式如下:

class Time
{
public:
  Time(int hour, int min, int sec)
    : _hour(hour)
    , _min(min)
    , _sec(sec)
  {}
private:
  int _hour;
  int _min;
  int _sec;
};

2 注意事项

对象初始化时,首先会根据初始化列表初始化成员变量(如果有自定义类型,会调用相应的默认构造函数),然后再执行函数体的内容。c++是根据初始化列表来知道怎么初始化成员变量的,如果没有显示的写初始化列表,函数会默认生成相应的初始化列表。

  1. 每个成员变量在初始化列表只能=出现一次(初始化只能出现一次)。
  2. 类中包含引用成员变量、const成员变量、自定义类型成员(并且该类没有默认构造函数时),必须将它们放在初始化列表进行初始化
  1. 成员变量在类中声明次序就是其在初始化列表中的初始化顺序,与其在初始化列表中的先后次序无关。

1~3

  1. 每个成员变量在初始化列表只能出现一次(初始化只能出现一次)。

类中包含引用成员变量、const成员变量、自定义类型成员(并且该类没有默认构造函数时),必须将它们

  1. 类中包含引用成员变量、const成员变量、自定义类型成员(并且该类没有默认构造函数时),必须将它们放在初始化列表进行初始化。

想要初始化自定义类型,首先要保证该类型存在默认构造函数(注意什么才是默认构造函数),如果没有会报错,因为当前类如果没有相应的构造函数,编译器会调用该类的默认构造函数,才能执行。代码如下:

对于自定义类型

class Date 
{
public:
  // 重点在构造函数,其他不是重点
  // 全缺省才是默认构造函数
  //Date(int year=0, int month=0, int day=0)
  //{
  //  _year = year;
  //  _month = month;
  //  _day = day;
  //}
  Date&  operator=(const Date& d1)
  {
    _year = d1._year;
    _month = d1._month;
    _day = d1._day;
    return *this;
  }
private:
  int _year;
  int _month;
  int _day;
};
class Time
{
public:
  Time(int hour=0, int min=0, int sec=0)
    : _hour(hour)
    , _min(min)
    , _sec(sec)
  {
  //在函数体初始化,比较麻烦,必须保证存在默认构造z函数
    Date t1;
    _d = t1;
  }
private:
  int _hour;
  int _min;
  int _sec;
  Date _d;
};
int main()
{
  Time t(6, 30, 30);
  return 0;
}


047cb0c0b3ee4b759828f27148538cb1.png

如果不显示写Date类的默认构造函数,那么编译器不会初始化_d的值,_d里面是随机值。反而如果把初始化的工作写在初始化列表,即便Date类里没有显示写默认构造函数,编译器也会将_d的自定义类型初始化为0.

class Date
{
public:
  Date& operator=(const Date& d1)
  {
    _year = d1._year;
    _month = d1._month;
    _day = d1._day;
    return *this;
  }
private:
  int _year;
  int _month;
  int _day;
};
class Time
{
public:
  Time(int hour = 0, int min = 0, int sec = 0)
    : _hour(hour)
    , _min(min)
    , _sec(sec)
    ,_d()
  {
  }
private:
  int _hour;
  int _min;
  int _sec;
  Date _d;
};
int main()
{
  Time t(6, 30, 30);
  return 0;
}


6c60281d28ec4375b7ab3598f8a2e706.png

对于const类型

const修饰的成员变量为常量,不可以在函数体内赋值,因为常量不可修改。但是可以在列表进行初始化,或者使用C++11的特性,在声明处赋值。


5d816e5c670e4993b343d01919141c6a.png

对于引用类型:

引用类型,引用必须在定义的时候初始化,并且不能重新赋值,所以也要写在初始化列表里面。


5a66f30dec2e4e1dad68333188a1a11d.png

  1. 成员变量在类中声明次序就是其在初始化列表中的初始化顺序,与其在初始化列表中的先后次序无关。

目录
相关文章
|
6天前
|
编译器 C++
【c++】构造函数赋值方式(初始化列表)
【c++】构造函数赋值方式(初始化列表)
|
6天前
|
编译器 C++
【c++】构造函数和析构函数
【c++】构造函数和析构函数
【c++】构造函数和析构函数
|
6天前
|
机器学习/深度学习 人工智能 算法
【动态规划】【组合数学】【C++算法】920播放列表的数量
【动态规划】【组合数学】【C++算法】920播放列表的数量
|
7天前
|
安全 编译器 C++
C++类与对象【对象的初始化和清理】
C++类与对象【对象的初始化和清理】
C++类与对象【对象的初始化和清理】
|
14天前
|
存储 编译器 C++
C++ 新特性 构造函数
C++ 新特性 构造函数
12 1
|
14天前
|
C语言 C++ 容器
C++新特性 初始化
C++新特性 初始化
14 0
|
15天前
|
编译器 C语言 C++
C++类和对象的细节原理:this指针、构造函数和析构函数、深浅拷贝、运算符重载、初始化列表、类的各种成员和方法
C++类和对象的细节原理:this指针、构造函数和析构函数、深浅拷贝、运算符重载、初始化列表、类的各种成员和方法
26 0
|
3天前
|
编译器 C语言 C++
【c++】类和对象4
【c++】类和对象4
23 2
|
3天前
|
编译器 C++
【c++】类和对象3
【c++】类和对象3
9 1
|
3天前
|
编译器 C++
【c++】类和对象2
【c++】类和对象2
8 1