C++ 初始化列表详解及实例代码

2020-01-06 16:13:26刘景俊


struct Test2
{
  Test1 test1 ;
  Test2(Test1 &t1):test1(t1){}
}

使用同样的调用代码,输出结果如下。

 C++,初始化列表,初始化列表详解,初始化列表详细介绍及示例代码

第一行输出对应 调用代码的第一行。第二行输出对应Test2的初始化列表,直接调用拷贝构造函数初始化test1,省去了调用默认构造函数的过程。所以一个好的原则是,能使用初始化列表的时候尽量使用初始化列表。

哪些东西必须放在初始化列表中

除了性能问题之外,有些时场合初始化列表是不可或缺的,以下几种情况时必须使用初始化列表

常量成员,因为常量只能初始化不能赋值,所以必须放在初始化列表里面 引用类型,引用必须在定义的时候初始化,并且不能重新赋值,所以也要写在初始化列表里面 没有默认构造函数的类类型,因为使用初始化列表可以不必调用默认构造函数来初始化,而是直接调用拷贝构造函数初始化。

对于没有默认构造函数的类,我们看一个例子。


struct Test1
{
  Test1(int a):i(a){}
  int i ;
};

struct Test2
{
  Test1 test1 ;
  Test2(Test1 &t1)
  {
    test1 = t1 ;
  }
};

以上代码无法通过编译,因为Test2类中Test1 test1;需要调用默认的构造函数,但是Test1类没有无参的构造函数,但是由于Test1没有默认的构造函数,故而编译错误。正确的代码如下,使用初始化列表代替赋值操作。


struct Test2
{
  Test1 test1 ;
  Test2(Test1 &t1):test1(t1){}
}

成员变量的初始化顺序

成员是按照他们在类中出现的顺序进行初始化的,而不是按照他们在初始化列表出现的顺序初始化的,看代码。


struct foo
{
  int i ;
  int j ;
  foo(int x):i(x), j(i){}; // ok, 先初始化i,后初始化j
};

再看下面的代码


struct foo
{
  int i ;
  int j ;
  foo(int x):j(x), i(j){} // i值未定义
};

这里i的值是未定义的,虽然j在初始化列表里面出现在i前面,但是i先于j定义,所以先初始化i,但i由j初始化,此时j尚未初始化,所以导致i的值未定义。所以,一个好的习惯是,按照成员定义的顺序进行初始化。

 感谢阅读,希望能帮助到大家,谢谢大家对本站的支持!


注:相关教程知识阅读请移步到C++教程频道。