文/楚雲風
基本解釋
const是一個C語言的關鍵字,它限定一個變量不允許被改變。使用const在一定程度上可以提高程序的健壯性,另外,在觀看別人代碼的時候,清晰理解const所起的作用,對理解對方的程序也有一些幫助。
雖然這聽起來很簡單,但實際上,const的使用也是c語言中一個比較微妙的地方,微妙在何處呢?請看下面幾個問題。
問題:const變量 & 常量
為什麼我像下面的例子一樣用一個const變量來初始化數組,ANSI C的編譯器會報告一個錯誤呢?
const int n = 5;
int 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巨集,這兩個都可以用來定義常量。
問題: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++是錯誤的。
問題:const變量 & 字串常量
請問下面的程式碼有什麼問題?
char *p = "i'm hungry!";
p[0]= 'I';
答案與分析:
上面的代碼可能會造成記憶體的非法寫入操作。分析如下, 「i'm hungry」實質上是字串常量,而常量往往被編譯器放在唯讀的記憶體區域,不可寫入的。p初始指向這個唯讀的記憶體區域,而p[0] = 'I'則企圖去寫這個地方,編譯器當然不會答應。
問題:const變量 & 字符串常量2
請問char a[3] = "abc" 合法嗎?使用它有什麼隱患?
答案與分析:
在標準C中這是合法的,但是它的生存環境非常狹小;它定義一個大小為3的數組,初始化為「abc」,注意,它沒有通常的字串終止符。
關鍵字const有什麼含意?
我只要一聽到被面試者說:「const意味著常數」
如果應試者能正確回答這個問題,我將問他一個附加的問題:
下面的聲明都是什麼意思?
const int a;
int const a;
const int *a;
int * const a;
int const * a const;
/******/
前兩個的作用是一樣,a是一個常量整數型態。第三個意味著a是一個指向
‧; 通過給最佳化產生器一些附加的信息,使用關鍵字const也許能產
‧; 合理地使用關鍵字const可以使編譯器很自然地保護那些不希望被
關鍵字volatile有什麼含意?並給出三個不同的例子。
一個定義為volatile的變量是說這變量可能會被意想不到地改
‧; 一個中斷服務子程序中會訪問到的非自動變量(Non
‧; 多執行緒應用中被幾個任務共享的變量
回答不出這個問題的人是不會被僱用的。我認為這是區分C程序員和嵌
假設被面試者正確地回答了這是問題(嗯,懷疑是否會是這樣)
‧; 一個指標可以是volatile 嗎?解釋為什麼。
‧; 下面的函數有什麼錯誤:
{
return *ptr * *ptr;
}
下面是答案:
‧; 是的。儘管這並不很常見。一個例子是當一個中服務子程序修該一個指
‧; 這段程式碼有點變態。這段程式碼的目的是用來返回指標*ptr指向值的平
{
int a,b;
a = *ptr;
b = *ptr;
return a * b;
}
由於*ptr的值可能被意想不到地該變,因此a和b可能是不同的
{
int a;
a = *ptr;
return a * a;
}