C++回顾之const对象、const成员函数、mutable类型

来源:互联网 发布:电脑编程语言 编辑:程序博客网 时间:2024/05/22 16:57

http://blog.csdn.net/ab198604/article/details/18980701

  先来总结一下const的一些普遍用法:

        1 可以用来定义常量,称为const常量

        const int x = 100; 

        2 用来修饰引用,称为const引用,表明不能修改引用所引用的变量的值

        const int &ref = x;

        int &ref2 = x; //Error, ref2不是常量,可以修改引用所指向的变量x的值,而x实际上是常量,不允许修改,故不合逻辑,这种用法是错误的

        3 用来修饰指针,有三种情况

        第一种,const出现在*左边,表示(*指针)是常量,而指针是变量,可以指向其它变量,如下:

        const int *p; 

        第二种,const 出现在*右边,表示指针是常量,指针不能指向其它变量,但是可以改变指针指向的变量的内容,如下:

        int *const p = &x; 

        第三种,const既在*左边,也在*右边,表明指针是常量,指针所指向的变量的内容也是常量,如下:

        const int *const p = &x;

        4 用来修饰类中的数据成员,称之为const数据成员,const数据成员的初始化要在构造函数的初始化列表中进行。不能在数据成员定义时直接初始化。

        5 用来修饰类对象,称为const对象

        const Test t(10);

        6 用来修饰类中的成员函数,称之为const成员函数,表示该函数不能修改对象的状态,也就是说它只能访问数据成员(Read Only),而不能修改数据成员。


        上面已经将const的所有用法进行了总结回顾,这里再分析一下,当const用来修饰类中的成员函数时,它不能修改类中的数据成员,但是在某些场合,它还是需要修改类的的某些数据成员时,该怎么办呢?这时候mutable类型的作用就发挥出来了,mutable的英文名是可变的,易变的,即由它修饰后的数据成员,无论是在const成员函数,还是非const成员函数均可以访问它,当然,除了static静态成员函数外(因为静态成员函数没有this指针,不能访问对象数据成员及函数)。这样一来,const成员函数就不能修改类中的非mutable修饰的数据成员,对这些数据成员只能是read only,但是却可以read/write那些由mutable修饰的数据成员了。在某些应用场合是比较灵活的。下面就举个简单的例子说明这些问题:

        

[cpp] view plaincopy在CODE上查看代码片派生到我的代码片
  1. #include <iostream>  
  2. using namespace std;  
  3.   
  4. class Test  
  5. {  
  6. public:  
  7.     Test(int x):x_(x), outputTimes_(0)  
  8.     {  
  9.     }  
  10.   
  11.     //因为Getx()这个成员函数不会去更改x数据成员,所以用const修饰,称之为const成员函数,它不能去修改数据成员的值  
  12.     int Getx() const  
  13.     {  
  14.         //x_ = 100 ;// Error, const成员函数不能修改数据成员的值  
  15.         cout << "const Getx..."<< endl;  
  16.         return x_;  
  17.     }  
  18.   
  19.     //非const成员函数与上面const成员函数构成重载  
  20.     int Getx()  //非const与const成员函数是可以构成重载的  
  21.     {  
  22.         cout << "Getx..."<<endl;  
  23.         return x_;  
  24.     }  
  25.   
  26.     int Setx()  
  27.     {  
  28.         return x_;  
  29.     }  
  30.   
  31.   
  32.     void Output() const   
  33.     {  
  34.         //x_ = 102; //Error,不能修改非mutable数据成员  
  35.         cout << "x = "<< x_ << endl;  
  36.         outputTimes_++; //可以修改mutable类型的数据成员  
  37.     }  
  38.       
  39.     int GetOutputTimes() const  
  40.     {  
  41.         return outputTimes_;  
  42.     }  
  43.   
  44. private:  
  45.     int x_;  
  46.     mutable int outputTimes_; //mutable数据成员  
  47.       
  48. };  
  49.   
  50. int main()  
  51. {  
  52.     //const对象,对象是常量,对象的状态不能更改,声明时要初始化  
  53.     const Test t(10);  
  54.     t.Getx(); //前提:若int Getx() const注释掉,则const对象是不能调用非const的Getx()成员函数的  
  55.      
  56.     //t.Setx(); //Error,理由同上,即const对象只能调用const成员函数,理由是:const对象是常量,对象的状态是不允许更改的,而非const即普通成员函数可以更改对象的状态,这所以为了实现不允许修改对象的状态这一目的,const对象禁止调用非const成员函数,而只能调用const成员函数,因为const成员函数也是禁止修改数据成员的,符合这一目的。  
  57.   
  58.     Test t2(20);  
  59.     t2.Getx(); //调用的是非const的Getx,因为t2是普通对象,不是const对象  
  60.     t2.Output(); //普通对象可以调用const成员函数,如果有重载的非const成员函数,则调用的是非const的成员函数  
  61.     t2.Output(); //每输出一次,outputTimes_加1,因为它是mutable类型的数据成员,允许在const成员函数中被修改  
  62.   
  63.     cout << "outputTimes= "<< t2.GetOutputTimes() << endl;   
  64.   
  65.     return 0;  
  66. }  

        这里需要注意的是const对象由于不能修改对象的状态,const对象只能调用const成员函数,而普通对象可以调用任何的成员函数。此外const对象在声明时要进行初始化。这与const修改的变量是一致的,如const int x = 100; 当类中定义了2个同名称的函数,其中一个用const修饰,另一个没有const修饰,他们之间是构成重载的,这时const对象调用的是const成员函数,而普通对象调用的是非const成员函数。


0 0
原创粉丝点击