C語言修飾詞之const

2021-05-28 02:53:21 字數 2929 閱讀 1202

const是乙個c語言的關鍵字,它限定乙個變數不允許被改變。使用const在一定程度上可以提高程式的安全性和可靠性,另外,在**別人**的時候,清晰理解const所起的作用,對理解對方的程式也有一些幫助。另外const在其他程式語言中也有出現,如c++、php5、c#.net、hc08 c。

c中const的使用

雖然這聽起來很簡單,但實際上,const的使用也是c語言中乙個比較微妙的地方,微妙在何處呢?請看下面幾個問題。

問題1:const變數 & 常量

為什麼下面的例子在使用乙個const變數來初始化陣列,ansi c的編譯器會報告乙個錯誤呢? 

const int n = 5;

nt a[n];

答案與分析:

1)、這個問題討論的是「常量」與「唯讀變數」的區別。常量肯定是唯讀的,例如5, "abc",等,肯定是唯讀的,因為常量是被編譯器放在記憶體中的唯讀區域,當然也就不能夠去修改它。而「唯讀變數」則是在記憶體中開闢乙個地方來存放它的值,只不過這個值由編譯器限定不允許被修改。c語言關鍵字const就是用來限定乙個變數不允許被改變的修飾符(qualifier)。上述**中變數n被修飾為唯讀變數,可惜再怎麼修飾也不是常量。而ansi c規定陣列定義時長度必須是「常量」,「唯讀變數」也是不可以的。

2)、注意:在ansi c中,這種寫法是錯誤的,因為陣列的大小應該是個常量,而const int n,n只是乙個變數(常量 != 不可變的變數,但在標準c++中,這樣定義的是乙個常量,這種寫法是對的),實際上,根據編譯過程及記憶體分配來看,這種用法本來就應該是合理的,只是 ansi c對陣列的規定限制了它。

3)、那麼,在ansi c 語言中用什麼來定義常量呢?答案是enum型別和#define巨集,這兩個都可以用來定義常量。

問題2:const變數 & const 限定的內容

下面的**編譯器會報乙個錯誤,請問,哪乙個語句是錯誤的呢? 

typedef char * pstr;

char string[4] = "abc";

const char *p1 = string;

const pstr p2 = string;

p1++;

p2++;

答案與分析:

問題出在p2++上。

1)、const使用的基本形式: const char m;

限定m不可變。

2)、替換1式中的m, const char *pm;

限定*pm不可變,當然pm是可變的,因此問題中p1++是對的。

3)、替換1式char, const newtype m;

限定m不可變,問題中的charptr就是一種新型別,因此問題中p2不可變,p2++是錯誤的。

問題3:const變數 & 字串常量

請問下面的**有什麼問題?

char *p = "i'm hungry!";

p[0]= 'i';

答案與分析:

上面的**可能會造成記憶體的非法寫操作。分析如下, "i'm hungry"實質上是字串常量,而常量往往被編譯器放在唯讀的記憶體區,不可寫。p初始指向這個唯讀的記憶體區,而p[0] = 'i'則企圖去寫這個地方,編譯器當然不會答應。

問題4:const變數 & 字串常量2

請問char a[3] = "abc" 合法嗎?使用它有什麼隱患?

答案與分析:

在標準c中這是合法的,但是它的生存環境非常狹小;它定義乙個大小為3的陣列,初始化為"abc",注意,它沒有通常的字串終止符'\0',因此這個陣列只是看起來像c語言中的字串,實質上卻不是,因此所有對字串進行處理的函式,比如strcpy、printf等,都不能夠被使用在這個假字串上。

問題5:const & 指標

型別宣告中const用來修飾乙個常量,有如下兩種寫法,那麼,請問,下面分別用const限定不可變的內容是什麼?

1)、const在前面

const int nvalue; //nvalue是const

const char *pcontent; //*pcontent是const, pcontent可變

const (char *) pcontent;//pcontent是const,*pcontent可變

char* const pcontent; //pcontent是const,*pcontent可變

const char* const pcontent; //pcontent和*pcontent都是const

2)、const在後面,與上面的宣告對等

int const nvalue; // nvalue是const

char const * pcontent;// *pcontent是const, pcontent可變

(char *) const pcontent;//pcontent是const,*pcontent可變

char* const pcontent;// pcontent是const,*pcontent可變

char const* const pcontent;// pcontent和*pcontent都是const

答案與分析:

乙個簡單的判斷方法:先忽略型別名(編譯器解析的時候也是忽略型別名),我們看const 離哪個近。「近水樓台先得月」,離誰近就修飾誰。

const int *p; //const 修飾*p,p 是指標,*p 是指標指向的物件,不可變

int const *p; //const修飾*p,p 是指標,*p 是指標指向的物件,不可變

int *const p; //const修飾p,p 不可變,p 指向的物件可變

const int *const p; //前乙個const 修飾*p,後乙個const 修飾p,指標p 和p 指向的物件都不可變 

關於修飾詞const

1.const修飾的唯讀變數必須在定義的同時初始化 因為如果沒有被賦初值,那麼變數被編譯器分配了乙個空間,並且賦給預設的乙個值,一般是很大的乙個負數,當我們再次給這個const變數賦值的時候就會出現錯誤,這個變數是唯讀的,一旦被賦值了就不能再改變了。2.case後面不可以是const修飾的唯讀變數,...

C語言修飾詞之violate

乙個定義為volatile的變數是說這變數可能會被意想不到地改變,這樣,編譯器就不會去假設這個變數的值了。精確地說就是,優化器在用到這個變數時必須每次都小心地重新讀取這個變數的值 from memory 而不是使用儲存在暫存器裡的備份。下面是volatile變數的幾個例子 1 並行裝置的硬體暫存器 ...

C語言修飾詞之violate

乙個定義為volatile的變數是說這變數可能會被意想不到地改變,這樣,編譯器就不會去假設這個變數的值了。精確地說就是,優化器在用到這個變數時必須每次都小心地重新讀取這個變數的值 from memory 而不是使用儲存在暫存器裡的備份。下面是volatile變數的幾個例子 1 並行裝置的硬體暫存器 ...