C語(yǔ)言中const的用法詳解
關(guān)注、星標(biāo)公眾號(hào),直達(dá)精彩內(nèi)容
來(lái)源:技術(shù)讓夢(mèng)想更偉大
作者:李肖遙
const基本介紹
const是constant的簡(jiǎn)寫(xiě),用來(lái)定義常量,它限定一個(gè)變量不允許被改變,產(chǎn)生靜態(tài)作用。const最開(kāi)始推出的目的是為了取代預(yù)編譯指令,取長(zhǎng)補(bǔ)短。
與define的對(duì)比
define是預(yù)編譯指令,定義的宏是在預(yù)處理階段展開(kāi)的,而const是普通變量的定義,是只讀變量,且是在編譯運(yùn)行階段使用的。
define定義的是常量,define定義的宏在編譯后消失了,它不占用內(nèi)存,而const定義的常變量本質(zhì)上仍然是一個(gè)變量,具有變量的基本屬性,有類型、占用存儲(chǔ)單元,除了不能作為數(shù)組的長(zhǎng)度,用const定義的常變量具有宏的優(yōu)點(diǎn),而且使用更方便。
define定義的對(duì)象沒(méi)有數(shù)據(jù)類型,編譯器只能機(jī)械地進(jìn)行字符替換,沒(méi)有類型安全檢查,即會(huì)出現(xiàn)“
邊際問(wèn)題”或者是“括號(hào)問(wèn)題”。而const定義的是變量,有數(shù)據(jù)類型。
下面我們從幾個(gè)方面來(lái)說(shuō)一下const的用法:
修飾局部變量
const int num=5;
int const num=5;
這兩種寫(xiě)法是一樣的,都是表示變量num的值不能被改變,用const修飾變量時(shí),一定要初始化,否則之后就不能再進(jìn)行賦值了(后面會(huì)講到一種特殊情況)。
接下來(lái)看看const用于修飾常量靜態(tài)字符串,例如:
const char* str="techdreamer";
如果沒(méi)有const的修飾,我們可能會(huì)在后不經(jīng)意間的修改代碼,比如str[4]=’D’,這樣會(huì)導(dǎo)致對(duì)只讀內(nèi)存區(qū)域的賦值,使程序異常終止。
而加上const修飾之后,這個(gè)錯(cuò)誤就能在程序被編譯的時(shí)候立即被檢查出來(lái),讓邏輯錯(cuò)誤在編譯期被發(fā)現(xiàn),避免我們?cè)诤罄m(xù)中繼續(xù)debug。
修飾全局變量
全局變量的作用域是整個(gè)文件,且全局變量的生存周期為程序運(yùn)行的整個(gè)過(guò)程,所以我們應(yīng)該盡量避免使用全局變量,一旦某個(gè)函數(shù)改變了全局變量的值,會(huì)影響到其他引用這個(gè)變量的函數(shù),是一個(gè)很隱蔽的操作。
如果一定要用全局變量,應(yīng)該盡量的使用const進(jìn)行修飾,防止不必要的人為修改,使用 const 修飾過(guò)的局部變量就有了靜態(tài)特性,它的生存周期也是程序運(yùn)行的整個(gè)過(guò)程,雖然有了靜態(tài)特性,但并不是說(shuō)它變成了靜態(tài)變量。
修飾常量指針與指針常量
常量指針
常量指針是指針指向的內(nèi)容是常量,可以有以下兩種定義方式。
const int * num;
int const * num;
以下兩點(diǎn)需要注意:
常量指針說(shuō)的是不能通過(guò)這個(gè)指針改變變量的值,但可以通過(guò)其他的引用來(lái)改變變量的值。
int cnt=5;
const int* num=&cnt;
cnt=6;
常量指針指向的值不能改變,但這并不意味著指針本身不能改變,常量指針可以指向其他的地址。
int cnt=5;
int tmp=6;
const int* num=&cnt;
num=&tmp;
指針常量
指針常量是指指針本身是個(gè)常量,不能再指向其他的地址,寫(xiě)法如下:
int *const num;
需要注意的是,指針常量指向的地址不能改變,但是地址中保存的數(shù)值是可以改變的,可以通過(guò)其他指向改地址的指針來(lái)修改。
int cnt=5;
int *tmp=&cnt;
int* const num=&cnt;
*tmp=6;
區(qū)分常量指針和指針常量的關(guān)鍵就在于星號(hào)的位置,我們以星號(hào)為分界線。
如果const在星號(hào)的左邊,則為常量指針
如果const在星號(hào)的右邊則為指針常量
如果我們將星號(hào)讀作‘指針’,將const讀作‘常量’的話,內(nèi)容正好符合。
int const * num;是常量指針,int *const num;是指針常量。
指向常量的常指針
還有一種情況是指向常量的常指針,這相當(dāng)于是常量指針與指針常量的結(jié)合,指針指向的位置不能改變并且也不能通過(guò)這個(gè)指針改變變量的值,例如
const int* const num;
這個(gè)代表num所指向的對(duì)象的值以及它的地址本身都不能被改變
修飾函數(shù)的形參
根據(jù)常量指針與指針常量,const修飾函數(shù)的參數(shù)也是分為三種情況
防止修改指針指向的內(nèi)容
void FUN(char *destin, const char *source);
其中 source 是輸入?yún)?shù),destin 是輸出參數(shù)。給 source 加上 const 修飾后,如果函數(shù)體內(nèi)的語(yǔ)句試圖改動(dòng) source 的內(nèi)容,編譯器將報(bào)錯(cuò),但反過(guò)來(lái)是可以的,編譯器允許將char *類型的數(shù)據(jù)賦值給const char *類型的變量。
防止修改指針指向的地址
void FUN ( int * const p1 , int * const p2 )
指針p1和指針p2指向的地址都不能修改。
以上兩種的結(jié)合。
在C語(yǔ)言標(biāo)準(zhǔn)庫(kù)中,有很多函數(shù)的形參都被 const 限制了,下面是部分函數(shù)的原型:
size_t strlen ( const char * str );
int strcmp ( const char * str1, const char * str2 );
char * strcat ( char * destination, const char * source );
char * strcpy ( char * destination, const char * source );
int system (const char* command);
int puts ( const char * str );
int printf ( const char * format, ... );
修飾函數(shù)的返回值
如果給以“指針傳遞”方式的函數(shù)返回值加 const 修飾,那么函數(shù)返回值(即指針)的內(nèi)容不能被修改,該返回值只能被賦給加const 修飾的同類型指針,例如
const char * FUN(void);
如下語(yǔ)句將出現(xiàn)編譯錯(cuò)誤:
char *str = FUN();
正確的用法是
const char *str = FUN();
思考
C與C++中的const用法有什么區(qū)別?
編譯器會(huì)給const定義的變量分配存儲(chǔ)空間嗎?
const變量能被其他文件extern引用嗎?
參考:C語(yǔ)言中文網(wǎng)
嵌入式編程專輯 Linux 學(xué)習(xí)專輯 C/C++編程專輯 Qt進(jìn)階學(xué)習(xí)專輯
關(guān)注我的微信公眾號(hào),回復(fù)“加群”按規(guī)則加入技術(shù)交流群。
點(diǎn)擊“閱讀原文”查看更多分享。
