天天看點

C++對const的用法筆記

 deprecated string 編譯器 linux gcc

Linux 環境下當GCC版本比較高時,編譯代碼可能出現的問題

問題是這樣産生的,先看這個函數原型:

void someFunc(char *someStr);

再看這個函數調用:

someFunc("I'm a string!");

把這兩個東西組合起來,用最新的g++編譯一下就會得到标題中的警告。

為什麼呢?原來char *背後的含義是:給我個字元串,我要修改它。

而理論上,我們傳給函數的字面常量是沒法被修改的。

是以說,比較和理的辦法是把參數類型修改為const char *。

這個類型說背後的含義是:給我個字元串,我隻要讀取它。

很自然的延伸一下。

如果我既要傳字面常量又要傳字元串變量怎麼辦呢?......重載

實驗:

對deprecated conversion from string constant to 'char *'此類警告的詳細解釋

假定你想使用一個char*類型的變量,有時指向一個字元串,有時指向另外一個字元串。開始的代碼就像這樣:

char *msg;

msg = "hello";

msg = "good-bye";

編譯器會對這段代碼給出兩段警示,說”deprecated conversion from string constant to 'char *'",意思就是說你沒有能力修改字元串的内容。如果将代碼寫成這樣,如:

char *msg = "hello";

*msg = 'j';

printf( "%s/n", "hello" );

編譯器會通過編譯,實際上會将msg指向的内容從"hello"轉變為"jello", 正确的解決方法是将msg聲明為一個指向不變字元串的指針:

const char  *msg;

msg = "hello";

msg = "good-bye";

這段代碼可以成功編譯,并且将msg指向的值如願改變,但如果你将指針指向的指進行指派:

*msg = 'j';

将會産生一個錯誤,不能修改一個字元串常量

注意如下的代碼,此代碼編譯時不會出現警告也不會出現任何錯誤:

const char  *msg;

char   buf[ 10 ];   //注意不能使用char *buf;

sprintf( buf, "%03d/n", 7 );

msg = buf;

改變buf的内容是可以的,因為它并沒有被聲明為常量。在這種情況下,msg将指向一個字元串,"007/n". 像這種語句

*buf = 'x';

将會正确編譯執行,但像

*msg = 'x';

将會産生一個警告,因為msg指向的内容不允許改變

還有一種方法是使用強制轉換,使用強制轉換意味着你清楚會出現什麼情況,不需要編譯器為你做出判斷,例如下面的代碼将不會産生警告:

char  *msg;

msg = (char *) "hello";

但一旦你使用強制轉換,編譯器對如下語句進行編譯時,也不會出現錯誤或警告

*msg = 'j';

這個錯誤将一直存在,但并不會被發現,直到運作時。那時再找出錯點就相當麻煩了,比編譯器提醒你,麻煩多了。是以,最好不要對字元串使用強制轉換。

Constant 指針

根據constant的位置不同,可以有以下四種情況:

const char* const msg_0;

const char       *msg_1;

char* const msg_2;

char       *msg_3;

其中,msg_0是一個constant指針指向一個const字元串。這個聲明編譯器會給出一個警告,因為msg_0的指向沒有被初始化,而且之後的語句也無法對mg_0進行指派,如

const char const *msg_0 = "hello";

會編譯成功,但

*msg_0 = 'j';或者

msg_0 = "good-bye"; 将會産生錯誤

msg_1既可以指向一個const字元串,也可以指向一個可變的字元串,但是不能修改它所指向的字元串的内容。

編譯msg_2這條語句,會出現和編譯msg_0一樣的錯誤。因為指針是一個常量,是以它應該首先被指派。如果剛開始已經指派,那麼它可以對指向的字元串内容進行修改,如:

char        buf[ 10 ];

char * const msg_2 = buf;

這段代碼裡,msg_2指向buf[0],并且永遠指向這個位址,不會改變;

對于msg_3,就沒太多可以說的。你可以改變指針,也可以改變指針指向的内容