说到构造函数,就必须要说到构造函数初始化列表。为什么要说它呢,下面来给各位客官娓娓道来。(目前有许多博客对此处已经说的很好了,可是沙米在前篇给大家写了构造函数的理解和应用场景博客后,还想让大家再深入的理解一下,故做此文章)

    构造函数进化流程:

    

java 类 构造函数初始化_java 类 构造函数初始化

    现在来详细聊聊:

    1、使用格式

    构造函数的初始化列表以冒号开头,后面跟着一系列以逗号分隔的初始化字段。

class Teacher
{
public:
	Teacher(int x):i(x),j(x){}; //初始化列表
private:
	int i;
	int j;
};

 

    2、  构造函数执行的两个阶段

    初始化阶段:所有类类型(class type)的成员都会在初始化阶段初始化,即使该成员没有出现在构造函数的初始化列表中。

    计算阶段:一般用于执行构造函数体内的赋值操作

    使用常规构造函数赋值类对象:

#include <iostream>
using namespace std; 

class Test_A
{
public:
	Test_A()
	{
		cout<<"构造函数Test_A()"<<endl;
	}

	Test_A(const Test_A& t1)
	{
		cout<<"拷贝构造函数Test_A()"<<endl;
		m_age = t1.m_age;
	}

	Test_A& operator = (const Test_A& t1)
	{
		cout<<"重载赋值运算符operator="<<endl;
		m_age = t1.m_age;
		return *this;
	}

	~Test_A()
	{
		cout<<"析构函数~Test_A()"<<endl;
	}
public:
	int m_age;
};

class Test_B
{
public:
	Test_B(Test_A& t1)
	{
		m_b = t1;
	}
public:
	Test_A m_b;
};

/*此函数相当于一个舞台,展示此函数内对象的完整生命周期*/
void display() 
{
	Test_A t1;
	Test_B t2(t1);
}

int main()
{
	display();
	system("pause");
	return 0;
}

 

    输出结果:

构造函数Test_A()
构造函数Test_A()
重载赋值运算符operator=
析构函数~Test_A()
析构函数~Test_A()

 

    从输出结果中可以看出,在执行Test_B t2(t1)的过程:

 (初始化阶段)

(计算阶段)

    使用初始化列表(只需修改类Test_B中的构造函数):

class Test_B
{
public:
	Test_B(Test_A& t1):m_b(t1){};   //使用了构造函数的初始化列表
public:
	Test_A m_b;
};

 

        输出结果:

构造函数Test_A()
拷贝构造函数Test_A()
析构函数~Test_A()
析构函数~Test_A()

 

     从输出结果中可以看出,执行Test_B t2(t1)的过程:

(初始化阶段)

(计算阶段)

java 类 构造函数初始化_java 类 构造函数初始化_02

     3、为什么需要——初始化列表

    (1)性能的提高,对于内置类型,使用初始化列表和构造函数内赋值性能差别不是很大,但是对于类类型来说,使用初始化列表,减少了一次计算阶段,如果是密集型类,效率将会更高。

    (2) 成员的类型是没有默认构造函数的类。若没有提供显示初始化式,则编译器隐式使用成员类型的默认构造函数,若类没有默认构造函数,则编译器尝试使用默认构造函数将会失败,必须使用初始化列表

    (3)const成员或引用类型的成员。因为const对象或引用类型只能初始化,不能对他们赋值,必须使用初始化列表

 注意:类中成员是按照他们在类中出现的顺序进行初始化的,而不是按照他们在初始化列表出现的顺序初始化的

 4、本质(很重要)

    在执行A类构造函数的初始化阶段,就将传入A类构造函数的参数值在A类的成员初始化时,进行了值得传递。

    沙米才疏学浅,希望大家多多给沙米提意见。,一起进步,提高。