常量函数、常量引用参数、常量引用返回值

常量函数、常量引用参数、常量引用返回值[C++]  

2009-12-01 09:42:05|  分类: 学术转载|字号 订阅

常量函数、常量引用参数、常量引用返回值[C++]
1. 关于常量引用
正像在C语言中使用指针一样,C++中通常使用引用 
有一个函数
... foo()
并且这个函数返回一个引用...
... & foo()
...., 一个指向位图(Bitmap)的引用 ...
Bitmap & foo()
.... 并且这个位图(bitmap)是常量
const Bitmap & foo ()

当然你也可以用指针来做同样的事情:
const Bitmap * foo()
foo 返回一个指针 ... 指向一个Bitmap ... 并有这个Bitmap是个常量.
Bitmap * const foo()
foo 返回某个东西,这个东西是常量 ... 这个东西又是指针 ... 一个指向Bitmap的指针.
const Bitmap * const foo()
foo 返回某个东西,这个东西是常量 ... 这个东西又是指针 ... 一个指向Bitmap的指针.
....
并且这个Bitmap也是常量.
指针常量与常量指针请参考Blog: http://www.cnblogs.com/JCSU/articles/1019219.html

const总是针对它左边的东西起作用, 唯一例外的是,如果const是一个最左边的标识符,那么const将针对它右边的东西起作用,因些 const int i; 与 int const i; 意思是相同的.
原文请参考: http://www.thescripts.com/forum/thread63796.html

2. 常量函数、常量引用参数、常量引用返回值
例1:bool verifyObjectCorrectness( const myObj &obj); //const reference parameter
例2:void Add(const int &arg)  const; //const function
例3: IStack const & GetStack() const { return _stack; } //return const reference

常量引用参数
本例中,一个myObj类型的对象obj通过引用传入函数verifyObjectCorrectness。为安全起见,使用了const关键字来确保函数verifyObjectCorrectness不会改变对象obj所引用的对象的状态。此外,通过声明参数常量,函数的使用者可以确保他们的对象不会被改变,也不必担心在调用函数时带来副作用。以下代码试图对声明为常量引用的形参进行修改,从而不会通过编译!
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 #include  < iostream > 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 using namespace std;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 class  Test
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 {
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 public :
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客     void  f(  const   int  &  arg);
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 private :
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客     int  value; 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客} ;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 void  Test::f(  const   int  &  arg)   常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 {
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客     arg= 10;  // 试图修改arg的值,此行将引起编译器错误 // error C2166: l-value specifies const object 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客     cout << " arg= " << arg << endl; 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客    value = 20 ;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客}   
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 void  main()
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 {
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客   int  i = 7 ;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  Test test;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  test.f(i);
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  cout << " i= " << i << endl; 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客} 常量函数 
1. 一个函数通过在其后面加关键字const,它将被声明为常量函数
2. 在C++,只有将成员函数声明为常量函数才有意义。带有const作后缀的常量成员函数又被称为视察者(inspector),没
    有const作后缀的非常量成员函数被称为变异者(mutator)
3. 与const有关的错误总是在编译时发现
4. [摘]If the function is not declared const, in can not be applied to a const object, and the compiler will  
         give an error message. A const function can be applied to a non-const object 
5. 在C++中,一个对象的所有方法都接收一个指向对象本身的隐含的this指针;常量方法则获取了一个隐含的常量this指针
    void func_name() const;
    以上说明函数func_name()不会改变*this。当你把this指针看成函数func_name()的一个不可见参数就理解了
    void func_name(T *this) (no const)
    void func_name(const T *this) (const)
6. 常量函数可以被任何对象调用,而非常量函数则只能被非常量对象调用,不能被常量对象调用,如: 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 class  Fred   常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 {
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  public :
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客    void  inspect()  const ;    //  This member promises NOT to change *this 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客     void  mutate();           //  This member function might change *this 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  } ;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  void  userCode(Fred &  changeable,  const  Fred &  unchangeable)
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客   常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 {
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客   changeable.inspect();    //  OK: doesn't change a changeable object 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客    changeable.mutate();     //  OK: changes a changeable object 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客   unchangeable.inspect();  //  OK: doesn't change an unchangeable object 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客    unchangeable.mutate();   //  ERROR: attempt to change unchangeable object 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  }  
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 #include  < iostream > 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 using namespace std;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客struct A
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 {
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客    void  f()  const   常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 { cout << " const function f is called " << endl; } 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客     void  f()   常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 { cout << " non-const function f is called " << endl; } 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客      void  g()   常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 { cout << " non-const function g is called " << endl; }  
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客} ;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 void  main()
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 {
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客    A a;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客      const  A &  ref_a  =  a;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客    a.f();  // calls void f() 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客     ref_a.f(); // calls void f () const 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客     a.g();  // ok, normal call 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客      ref_a.g();  //  error, const object can not call non-const function 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 } 7. 在类中允许存在同名的常量函数和非常量函数,编译器根据调用该函数的对象选择合适的函数
    当非常量对象调用该函数时,先调用非常量函数;
    当常量对象调用该函数时,只能调用常量函数;
    如果在类中只有常量函数而没有与其同名的非常量函数,则非常量与常量对象都可调用该常量函数;如:     常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 #include  < iostream > 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 using namespace std;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客struct A
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 {
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客     void   f()  const    常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 { cout << " const function f is called " << endl; } 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客     void   f()   常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 { cout << " non-const function f is called " << endl; } 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客} ;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 void  main()
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 {
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客    A a;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客     const  A &  ref_a  =  a;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客    a.f(); // calls void f()
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客     ref_a.f(); //  calls void f () const 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 } 输出结果:
non-const function f is called
const function f is called 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 #include  < iostream > 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 using namespace std;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客struct A  常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 {
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客     void   f()  const    常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 { cout << " const function f is called " << endl; } 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客} ;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 void  main()
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 {
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客    A a;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客     const  A &  ref_a  =  a;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客    a.f();  //  calls void f() const 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客     ref_a.f(); //  calls void f () const 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 } 输出结果:
const function f is called
const function f is called

8. 以下代码试图修改类的数据成员,引起编译错误 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 #include  < iostream > 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 using namespace std;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 class  Test
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 {
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 public :
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客     void  f( const   int &  arg)   const ;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 private :
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客      int  value; 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客} ;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 void  Test::f( const   int &  arg)   const  常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 {
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客     // arg=10;  
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客     cout << " arg= " << arg << endl; 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客     value = 20 ;  // 试图修改Test的数据成员,此行将引起编译器错误  // error C2166: l-value specifies const object 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 }   
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 void  main()
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客 {
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客   int  i = 7 ;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  Test test;
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  test.f(i);
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客  cout << " i= " << i << endl; 
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客
9. const关键字不能用在构造函数与析构函数中。因为构造函数的目的是初始化域值,因此它必须更改对象,析构函数同理

常量引用返回值
如果你想从常量方法(函数)中通过引用返回this对象的一个成员, 你应该使用常量引用来返回它,即const X&
也就是说你想通过引用返回的东西如果从逻辑上来讲是this对象的一部分(与它是否在物理上嵌入在this对象中无关),那么常量方法需要通过常量引用或者通过值来返回,而不能通过非常量引用返回
原文请参考: http://www.parashift.com/c++-faq-lite/const-correctness.html#faq-18.10
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客class Person 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客{
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客public:
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客    const string& name_good() const; // Right: the caller can't change the name
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客    string& name_evil() const;       // Wrong: the caller can change the name
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客    常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客.
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客};
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客void myCode(const Person& p) // You're promising not to change the Person object常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客{
常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客    p.name_evil() = "Igor";  // 常量函数、常量引用参数、常量引用返回值[C++] - 风殇 - 张霁的博客but you changed it anyway!!

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值