• C++类的初始化


    转载自:http://blog.csdn.net/coder_xia/article/details/7447822

    1、关于构造函数

    1)用构造函数确保初始化

    对于一个空类

    1. class Empty { };  

    编译器会自动声明4个默认函数:构造函数,拷贝构造函数,赋值函数,析构函数(当然,如果不想使用自动生成的函数,就应该明确拒绝),这些生成的函数都是public且inline。构造函数对数据成员进行初始化,使用未初始化值可能导致无法预知的错误,所以,确保每一个构造函数都将每一个成员初始化。

    2)为什么构造函数不能有返回值

    如果有返回值,要么编译器必须知道怎么处理返回值,要么就客户程序员显式调用构造函数和析构函数,这样,还有安全性么?

    3)为什么构造函数不能为虚函数

    简单来说,虚函数调用的机制,是知道接口而不知道其准确对象类型的函数,但是创建一个对象,必须知道对象的准确类型;当一个构造函数被调用时,它做的首要事情之一就是初始化它的VPTR来指向VTABLE。

    4)构造函数的一个面试题:

    1. #include <iostream>  
    2. using namespace std;  
    3.   
    4. class Base   
    5. {  
    6. private:  
    7.     int i;  
    8. public:  
    9.     Base(int x)  
    10.     {  
    11.         i = x;  
    12.     }  
    13. };  
    14.   
    15. class Derived : public Base  
    16. {  
    17. private:  
    18.     int i;  
    19. public:  
    20.     Derived(int x, int y)  
    21.     {  
    22.         i = x;  
    23.     }  
    24.     void print()  
    25.     {  
    26.         cout << i + Base::i << endl;  
    27.     }  
    28. };  
    29.   
    30. int main()  
    31. {  
    32.     Derived A(2,3);  
    33.     A.print();  
    34.     return 0;  
    35. }  

    首先,是访问权限问题,子类中直接访问Base::i是不允许的,应该将父类的改为protected或者public(最好用protected)

    其次,统计父类和子类i的和,但是通过子类构造函数没有对父类变量进行初始化;此处编译会找不到构造函数,因为子类调用构造函数会先找父类构造函数,但是没有2个参数的,所以可以在初始化列表中调用父类构造函数

    最后个问题,是单参数的构造函数,可能存在隐式转换的问题,因为单参数构造函数,和拷贝构造函数形式类似,调用时很可能会发生隐式转换,应加上explicit关键字,修改后如下(程序员面试宝典上只改了前2个)

    1. #include <iostream>  
    2. using namespace std;  
    3.   
    4. class Base   
    5. {  
    6. protected:  
    7.     int i;  
    8. public:  
    9.     explicit Base(int x)  
    10.     {  
    11.         i = x;  
    12.     }  
    13. };  
    14.   
    15. class Derived : public Base  
    16. {  
    17. private:  
    18.     int i;  
    19. public:  
    20.     Derived(int x, int y):Base(x)  
    21.     {  
    22.         i = y;  
    23.     }  
    24.     void print()  
    25.     {  
    26.         cout << i + Base::i << endl;  
    27.     }  
    28. };  
    29.   
    30. int main()  
    31. {  
    32.     Derived A(2,3);  
    33.     A.print();  
    34.     return 0;  
    35. }  

    2、初始化列表

    1)使用初始化列表提高效率

    常用的初始化可能如下:

    1. class Student   
    2. {  
    3. public:  
    4.     Student(string in_name, int in_age)  
    5.     {  
    6.         name = in_name;  
    7.         age = in_age;  
    8.     }  
    9. private :  
    10.     string name;  
    11.     int    age;  
    12. };  

        以前楼主也习惯这么写,可以达到预期效果,不过不是最佳做法,因为在构造函数中,是对name进行赋值,不是初始化,而string对象会先调用它的默认构造函数,再调用string类(貌似是basic_string类)的赋值构造函数;对于上例的age,因为int是内置类型,应该是赋值的时候获得了初值。

        要对成员进行初始化,而不是赋值,可以采用初始化列表(member initialization list)改写为如下:

    1. class Student   
    2. {  
    3. public:  
    4.     Student(string in_name, int in_age):name(in_name),age(in_age) {}  
    5. private :  
    6.     string name;  
    7.     int    age;  
    8. };  

        结果与上例相同,不过在初始化的时候调用的是string的拷贝构造函数,而上例会调用两次构造函数,从性能上会有不小提升

        有的情况下,是必须使用初始化列表进行初始化的:const对象、引用对象

    2)初始化列表初始顺序

    考虑以下代码:

    1. #include <iostream>  
    2. using namespace std;  
    3.   
    4. class Base   
    5. {  
    6. public:  
    7.     Base(int i) : m_j(i), m_i(m_j) {}  
    8.     Base() : m_j(0), m_i(m_j) {}  
    9.     int get_i() const  
    10.     {  
    11.         return m_i;  
    12.     }  
    13.     int get_j() const  
    14.     {  
    15.         return m_j;  
    16.     }  
    17.   
    18. private:  
    19.     int m_i;  
    20.     int m_j;  
    21.   
    22. };  
    23.   
    24. int main()  
    25. {  
    26.     Base obj(98);  
    27.     cout << obj.get_i() << endl << obj.get_j() << endl;  
    28.     return 0;  
    29. }  

         
    输出为一个随机数和98,为什么呢?因为对于初始化列表而言,对成员变量的初始化,是严格按照声明次序,而不是在初始化列表中的顺序进行初始化,如果改为赋值初始化则不会出现这个问题,当然,为了使用初始化列表,还是严格注意声明顺序吧,比如先声明数组大小,再声明数组这样。

    1)使用类的初始化可以提高效率:

     1 #include<iostream>
     2 #include<stdio.h>
     3 using namespace std;
     4 class boy {
     5       public:
     6   boy(string name,int age):name(name),age(age)//类的初始化
     7       {
     8       cout<<this->name<<endl;
     9       cout<<this->age<<endl;
    10       }
    11       ~boy()
    12       {
    13           cout<<this->name<<endl<<this->age<<endl;
    14       }
    15       string name;
    16       int age;
    17 };
    18 class desk{
    19 
    20 public:
    21       desk(int x):w(x),h(w){}
    22       int h;
    23       int w;
    24 };
    25 int main()
    26 {
    27     boy a("abc",22);
    28     desk b(25);
    29 cout<<b.h<<endl<<b.w<<endl;//类的初始化顺序按照声明的顺序
    30     return 0;
    31 
    32   }

     

  • 相关阅读:
    【女农场主手记】漂亮地晕和扯淡地安全
    2008最新个人所得税计算公式
    oracle 字符集乱码解决 详细解释了oracle字符集的问题
    单反相机的传奇—佳能单反50年辉煌之路(连载三)
    单反相机的传奇—佳能单反50年辉煌之路(连载七)
    2009年5月重上涠洲岛,大吃海鲜
    单反相机的传奇—佳能单反50年辉煌之路(连载六)
    单反相机的传奇—佳能单反50年辉煌之路(连载八)
    单反相机的传奇—佳能单反50年辉煌之路(连载二)
    单反相机的传奇—佳能单反50年辉煌之路(连载十)
  • 原文地址:https://www.cnblogs.com/cs1003/p/2809318.html
Copyright © 2020-2023  润新知