1. 程式人生 > >C++const用法淺析

C++const用法淺析

Const用法淺析

1. const修飾普通變數和指標

const修飾變數,一般有兩種寫法:

const TYPE value;

TYPE 
const value;

這兩種寫法在本質上是一樣的。它的含義是:const修飾的型別為TYPE的變數value是不可變的。

對於一個非指標的型別TYPE,無論怎麼寫,都是一個含義,即value只不可變。

例如:

constint nValue;         //nValue是constintconst nValue;    // nValue是const
但是對於指標型別的TYPE,不同的寫法會有不同情況,例如:

A. 
constchar*pContent;

B. 
char

*const pContent;

C. 
charconst*pContent;

D. 
constchar*const pContent;


對於前三種寫法,我們可以換個方式,給其加上括號

A. 
const (char*pContent;

B. (
char*const pContent;

C. (
charconst*pContent;

這樣就一目瞭然。根據對於const修飾非指標變數的規則,很明顯,A
=C.


- 對於A,C, const修飾的型別為char的變數*pContent為常量,因此,pContent的內容為常量不可變.

- 對於B, 其實還有一種寫法: const (char*) pContent;

含義為:const修飾的型別為char
*的變數pContent為常量,因此,pContent指標本身為常量不可變.

- 對於D, 其實是A和B的混合體,表示指標本身和指標內容兩者皆為常量不可變


總結:

(
1) 指標本身是常量不可變

(
char*const pContent;

const (char*) pContent;


(
2) 指標所指向的內容是常量不可變

const (char*pContent;

(
charconst*pContent;


(
3) 兩者都不可變

constchar*const pContent;


還有其中區別方法:

沿著
*號劃一條線,

如果const位於
*的左側,則const就是用來修飾指標所指向的變數,即指標指向為常量;

如果const位於
*的右側,const就是修飾指標本身,即指標本身是常量。

2. const修飾函式引數

const修飾函式引數是它最廣泛的一種用途,它表示函式體中不能修改引數的值(包括引數本身的值或者引數其中包含的值)。它可以很好

void function(constint Var); //傳遞過來的引數在函式內不可以改變(無意義,因為Var本身就是形參)void function(constchar* Var); //引數指標所指內容為常量不可變void function(char*const Var); //引數指標本身為常量不可變(也無意義, 因為char* Var也是形參)
 

引數為引用,為了增加效率同時防止修改。

修飾引用引數時:

void function(const Class& Var);//引用引數在函式內不可以改變void function(const TYPE& Var); //引用引數在函式內為常量不可變3const 修飾函式返回值

const修飾函式返回值其實用的並不是很多,它的含義和const修飾普通變數以及指標的含義基本相同。

(
1constint fun1() 這個其實無意義,因為引數返回本身就是賦值。

(
2constint* fun2() 

呼叫時 
constint*pValue = fun2();

我們可以把fun2()看作成一個變數,那麼就是我們上面所說的1.(
1)的寫法,即指標內容不可變。

(
3int*const fun3() 

呼叫時 
int*const pValue = fun2();

我們可以把fun2()看作成一個變數,那麼就是我們上面所說的1.(
2)的寫法,即指標本身不可變。 4. const修飾類物件/物件指標/物件引用

const修飾類物件表示該物件為常量物件,其中的任何成員都不能被修改。對於物件指標和物件引用也是一樣。

const修飾的物件,該物件的任何非const成員函式都不能被呼叫,因為任何非const成員函式會有修改成員變數的企圖。

例如:

class AAA

{
   
void func1();

void func2() const;

}

const AAA aObj;

aObj.func1(); ×

aObj.func2(); 正確

 

const AAA* aObj =new AAA();

aObj
->func1(); ×

aObj
->func2(); 正確

 
5. const修飾成員變數

const修飾類的成員函式,表示成員常量,不能被修改,同時它只能在初始化列表中賦值。


class A

{

   …

   
constint nValue;       //成員常量不能被修改
   …

   A(
int x): nValue(x) {}; //只能在初始化列表中賦值
}

 
6. const修飾成員函式

const修飾類的成員函式,則該成員函式不能修改類中任何非const成員函式。一般寫在函式的最後來修飾。

 

class A

{

   …

void function()const//常成員函式, 它不改變物件的成員變數. 也不能呼叫類中任何非const成員函式。
}

對於const類物件
/指標/引用,只能呼叫類的const成員函式,因此,const修飾成員函式的最重要作用就是限制對於const物件的使用。

const成員函式應該在函式原型說明和函式定義中都增加const限定:

//標頭檔案xxx.h

class XXX
{
 void doit() const; 

 void doit(); 

};

//實現檔案xxx.cpp

#include "xxx.h"

void XXX::doit() const
{
 int i = 0;
}
void XXX::doit()
{
 int i = 0;
}

7. const常量與define巨集定義的區別

(
1) 編譯器處理方式不同

define巨集是在預處理階段展開。

const常量是編譯執行階段使用。

(
2) 型別和安全檢查不同

define巨集沒有型別,不做任何型別檢查,僅僅是展開。

const常量有具體的型別,在編譯階段會執行型別檢查。

(
3) 儲存方式不同

define巨集僅僅是展開,有多少地方使用,就展開多少次,不會分配記憶體。

const常量會在記憶體中分配(可以是堆中也可以是棧中)。

// Test.cpp : Defines the entry point for the console application.

//

#include "stdafx.h"

#include <iostream>

class XXX

{

public:

void executeConstMethod() const

{

std::cout<<"XXX::executeConstMethod()/n";

}

void executeNonConstMethod()

{

std::cout<<"XXX::executeNonConstMethod()/n";

}

};

class Test

{

public:

const int constInt; //only can be initialize in class constructor

int nonConstInt;

const XXX constMember;

XXX nonConstMember;

Test(int i):constInt(i)

{

}

/************************************************************************

* in a const method, only const member method could be directly called.

* non-const member could be called, as std::cout<<nonConstInt;

* non-const member's non-const member method could not be called.

************************************************************************/

void executeConstMethod() const

{

constMember.executeConstMethod();

/* Compile Error: cannot convert 'this' pointer from 'const XXX' to 'XXX &' */

//constMember.executeNonConstMethod();

/* Compile Error: cannot convert 'this' pointer from 'const Test' to 'Test &' */

//executeNonConstMethod();

anotherConstMethod(); // works well.

nonConstMember.executeConstMethod(); //

std::cout<<nonConstInt; // works well too.

}

void anotherConstMethod() const

{

std::cout<<"Test::anotherConstMethod()/n";

}

void executeNonConstMethod()

{

// no restriction in non-const method.

constMember.executeConstMethod();

/* const method only can call const its const method. */

//constMember.executeNonConstMethod();

nonConstMember.executeConstMethod();

nonConstMember.executeNonConstMethod();

}

};

int _tmain(int argc, _TCHAR* argv[])

{

Test test(4);

/* non const variant can call both const and non-const method. */

test.executeNonConstMethod();

test.executeConstMethod();

getchar();

return 0;

}