C++中构造函数初始化列表与构造函数体的区别

时间:2022-09-09 16:28:02

既然构造函数初始化列表和构造函数体内赋值都可以对成员变量赋值,那么二者有何区别,是不是二者等效呢? 


构造函数可以分两个阶段进行:(1)初始化阶段;(2)普通计算阶段。计算阶段也就是由函数体内所有的语句组成。不管成员是否在构造函数初始化列表中显式初始化,类的数据成员初始化总是在初始化阶段进行,初始化阶段先于计算阶段。构造函数初始化列表是对类的成员做初始化,而在构造函数体内只是对类的数据成员进行了一次赋值操作。

构造函数初始化列表只是指定了成员的初始值,并没有指定初始化顺序,那么成员初始化顺序又是怎样的呢?成员的初始化顺序就是定义成员的顺序,第一个定义的成员首先被初始化,然后是第二个等等。


一、若类的数据成员是静态的(const)和引用类型,必需用初始化列表 

静态(const)的数据成员只能初始化而不能赋值,同样引用类型也是只可以被初始化,那么只有用初始化列表。 

如: 
C++代码  C++中构造函数初始化列表与构造函数体的区别
  1. #include <iostream>  
  2. #include <string>  
  3. using namespace std;  
  4.   
  5. template<class t>  
  6. class namedptr {  
  7. public:  
  8.     namedptr(const string& initname, t *initptr);  
  9. private:  
  10.     const string name; //静态数据成员的初始化必需用初始化列表  
  11.     t * const ptr;  
  12. };  
  13.   
  14.   
  15.   
  16. template<class t>  
  17. namedptr<t>::namedptr(const string& initname, t *initptr  )  
  18. : name(initname), ptr(initptr)  
  19. {}  
  20.   
  21. //第二种方法是在构造函数体内赋值:  
  22.   
  23. //template<class t>  
  24. //namedptr<t>::namedptr(const string& initname, t *initptr)  
  25. //{  
  26. //  name = initname;  
  27. //  ptr = initptr;  
  28. //}  
  29.   
  30.   
  31. int main()  
  32. {  
  33.     int a  = 10;  
  34.     namedptr<int> Test("SHENZHEN",&a);  
  35. }  

由于 const string name; //静态数据成员的初始化必需用初始化列表 
t * const ptr; 
是静态的,如果用构造函数体内赋值,编译会出错。 

二、构造函数体内赋值会带来额外的开销,效率会低于构造函数初始化列表 

上面的例子改一改: 

C++代码  C++中构造函数初始化列表与构造函数体的区别
  1. template<class t>  
  2. class namedptr {  
  3. public:  
  4.     namedptr(const string& initname, t *initptr);  
  5. private:  
  6.     string name; //静态数据成员的初始化必需用初始化列表  
  7.     t * ptr;  
  8. };  


并且用这两中初始化方法做对比: 
C++代码  C++中构造函数初始化列表与构造函数体的区别
  1. //第一种方法:初始化列表  
  2. template<class t>  
  3. namedptr<t>::namedptr(const string& initname, t *initptr  )  
  4. : name(initname), ptr(initptr)  
  5. {}  
  6.   
  7. //第二种方法是在构造函数体内赋值:  
  8.   
  9. template<class t>  
  10. namedptr<t>::namedptr(const string& initname, t *initptr)  
  11. {  
  12.     name = initname;  
  13.     ptr = initptr;  
  14. }  


当用第二种方法初始化数据成员时会两次对string的成员函数的调用:一次是缺省构造函数,另一次是赋值。 

而用第一种方法(初始化列表)只是一次调用缺省的构造函数,并不会调用赋值函数。会减少不必要的开支,当类相当复杂时,就会看出使用初始化列表的好处。


hzhsan注:区别不只上面这些!!!