假定你想使用一個(gè)char*類(lèi)型的變量,有時(shí)指向一個(gè)字符串,有時(shí)指向另外一個(gè)字符串。開(kāi)始的代碼就像這樣:
char *msg;
msg = "hello";
msg = "good-bye";
編譯器會(huì )對這段代碼給出兩段警示,說(shuō)”deprecated conversion from string constant to 'char *'",意思就是說(shuō)你沒(méi)有能力修改字符串的內容。如果將代碼寫(xiě)成這樣,如:
char *msg = "hello";
*msg = 'j';
printf( "%s\n", "hello" );
編譯器會(huì )通過(guò)編譯,實(shí)際上會(huì )將msg指向的內容從"hello"轉變?yōu)?jello", 正確的解決方法是將jmsg聲明為一個(gè)指向不變字符串的指針:
const char *msg;
msg = "hello";
msg = "good-bye";
這段代碼可以成功編譯,并且將msg指向的值如愿改變,但如果你將指針指向的指進(jìn)行賦值:
*msg = 'j';
將會(huì )產(chǎn)生一個(gè)錯誤,不能修改一個(gè)字符串常量
注意如下的代碼,此代碼編譯時(shí)不會(huì )出現警告也不會(huì )出現任何錯誤:
const char *msg;
char buf[ 10 ];
sprintf( buf, "%03d\n", 7 );
msg = buf;
改變buf的內容是可以的,因為它并沒(méi)有被聲明為常量。在這種情況下,msg將指向一個(gè)字符串,"007\n". 像這種語(yǔ)句
*buf = 'x';
將會(huì )正確編譯執行,但像
*msg = 'x';
將會(huì )產(chǎn)生一個(gè)警告,因為msg指向的內容不允許改變
還有一種方法是使用強制轉換,使用強制轉換意味著(zhù)你清楚會(huì )出現什么情況,不需要編譯器為你做出判斷,例如下面的代碼將不會(huì )產(chǎn)生警告:
char *msg;
msg = (char *) "hello";
但一旦你使用強制轉換,編譯器對如下語(yǔ)句進(jìn)行編譯時(shí),也不會(huì )出現錯誤或警告
*msg = 'j';
這個(gè)錯誤將一直存在,但并不會(huì )被發(fā)現,直到運行時(shí)。那時(shí)再找出錯點(diǎn)就相當麻煩了,比編譯器提醒你,麻煩多了。所以,最好不要對字符串使用強制轉換。
Constant 指針
根據constant的位置不同,可以有以下四種情況:
const char* const msg_0;
const char *msg_1;
char* const msg_2;
char *msg_3;
其中,msg_0是一個(gè)constant指針指向一個(gè)const字符串。這個(gè)聲明編譯器會(huì )給出一個(gè)警告,因為msg_0的指向沒(méi)有被初始化,而且之后的語(yǔ)句也無(wú)法對mg_0進(jìn)行賦值,如
const char const *msg_0 = "hello";
會(huì )編譯成功,但
*msg_0 = 'j';或者
msg_0 = "good-bye"; 將會(huì )產(chǎn)生錯誤
msg_1既可以指向一個(gè)const字符串,也可以指向一個(gè)可變的字符串,但是不能修改它所指向的字符串的內容。
編譯msg_2這條語(yǔ)句,會(huì )出現和編譯msg_0一樣的錯誤。因為指針是一個(gè)常量,所以它應該首先被賦值。如果剛開(kāi)始已經(jīng)賦值,那么它可以對指向的字符串內容進(jìn)行修改,如:
char buf[ 10 ];
char const *msg_2 = buf;
這段代碼里,msg_2指向buf[0],并且永遠指向這個(gè)地址,不會(huì )改變;
對于msg_3,就沒(méi)太多可以說(shuō)的。你可以改變指針,也可以改變指針指向的內容