C指针5:字符串指针(指向字符串的指针)
在C語言中沒有特定的字符串類型,我們通常是將字符串放在一個(gè)字符數(shù)組中,字符數(shù)組實(shí)際上是一系列字符的集合,也就是字符串(String)。在C語言中,沒有專門的字符串變量,沒有string類型,通常就用一個(gè)字符數(shù)組來存放一個(gè)字符串。C語言規(guī)定,可以將字符串直接賦值給字符數(shù)組。
例如:
char str[30] = {"c.zhongguo.com"};//可以使用str[i]來訪問元素
char str[30] = "c.zhongguo.com"; //這種形式更加簡潔,實(shí)際開發(fā)中常用//也可以不指定長度
數(shù)組第 0 個(gè)元素為'c',第 1 個(gè)元素為'.',第 2 個(gè)元素為'z',后面的元素以此類推。
首先明確一點(diǎn),同數(shù)組一樣字符串中的所有字符在內(nèi)存中是連續(xù)排列的。
先觀察一下下邊的輸出:
#include <stdio.h>
#include <iostream>
#include <string.h>
using namespace std;int main() {char str[] = "http://c.biancheng.net";char *pstr = str;//將字符串首地址給它c(diǎn)out << str << endl;//輸出http://c.biancheng.netprintf(str);輸出http://c.biancheng.net//最正規(guī)的是使用上格式控制符%s輸出字符串,%p輸出地址。return 0;
}
結(jié)果展示:
總結(jié):字符數(shù)組歸根結(jié)底還是一個(gè)數(shù)組,上節(jié)講到的關(guān)于指針和數(shù)組的規(guī)則同樣也適用于字符數(shù)組。
#include <stdio.h>
#include <iostream>
#include <string.h>
using namespace std;int main() {char str[] = "china.zhongguo.com";//未定義長度//p 指向的數(shù)組元素是 char 類型,所以 p 的類型必須也是char *char *pstr = str;//定義一個(gè)指針變量pstr,該指針變量初始化為str字符數(shù)組即字符串的首地址。int len = strlen(str), i;//使用*(pstr+i)for (i = 0; i<len; i++) {printf("%c", *(pstr + i));//先地址加再取元素}printf("\n");//使用pstr[i] 就是上節(jié)課里邊的p[i]即地址[i]for (i = 0; i<len; i++) {printf("%c", pstr[i]);}printf("\n");//使用*(str+i)for (i = 0; i<len; i++) {printf("%c", *(str + i));}printf("\n");return 0;
}
結(jié)果展示:
----------以上是使用字符數(shù)組來表示字符串,所以除了字符數(shù)組,C語言還支持另外一種表示字符串的方法,就是直接使用一個(gè)指針指向字符串,如下:(此處為什么直接將字符串賦給指針變量,需要不是應(yīng)該是地址嗎?,規(guī)定直接使用一個(gè)指針指向字符串)
char *str = "ABC//DEF.COM";
//或者
char *str;
str = "http://c.biancheng.net";//此處為什么直接將字符串賦給指針變量
字符串中的所有字符在內(nèi)存中是連續(xù)排列的,str 指向的是字符串的第 0 個(gè)字符;我們通常將第 0 ?個(gè)字符的地址稱為字符串的首地址。字符串中每個(gè)字符的類型都是char,所以 str 的類型也必須是char *。
#include <stdio.h>
#include <iostream>
#include <string.h>
using namespace std;int main() {char *str = "ABC//DEF.COM";int len = strlen(str), i;//直接輸出字符串printf("%s\n", str);//使用*(str+i)for (i = 0; i<len; i++) {printf("%c", *(str + i));//地址先加再取元素}printf("\n");//使用str[i]for (i = 0; i<len; i++) {printf("%c", str[i]);//使用地址[i]取數(shù)據(jù)}printf("\n");return 0;
}
結(jié)果展示:
這樣去取元素的方式和字符數(shù)組是非常相似,它們都可以使用%s輸出整個(gè)字符串,都可以使用*或[ ]獲取單個(gè)字符,但是這兩種表示字符串的方式還是有區(qū)別的,它們最根本的區(qū)別是在內(nèi)存中的存儲(chǔ)區(qū)域不一樣,字符數(shù)組存儲(chǔ)在全局?jǐn)?shù)據(jù)區(qū)或棧區(qū),第二種形式的字符串存儲(chǔ)在常量區(qū)。全局?jǐn)?shù)據(jù)區(qū)和棧區(qū)的字符串(也包括其他數(shù)據(jù))有讀取和寫入的權(quán)限,而常量區(qū)的字符串(也包括其他數(shù)據(jù))只有讀取權(quán)限,沒有寫入權(quán)限。
內(nèi)存權(quán)限的不同導(dǎo)致的一個(gè)明顯結(jié)果就是,字符數(shù)組在定義后可以讀取和修改每個(gè)字符,而對于第二種形式的字符串,一旦被定義后就只能讀取不能修改,任何對它的賦值都是錯(cuò)誤的。
我們將第二種形式的字符串稱為字符串常量,意思很明顯,常量只能讀取不能寫入。請看下面:
#include <stdio.h>
int main()
{char *str = "Hello China!";str = "I love C!"; //正確str[3] = 'P'; //錯(cuò)誤return 0;
}
這段代碼能夠正常編譯和鏈接,但在運(yùn)行時(shí)會(huì)出現(xiàn)段錯(cuò)誤(Segment Fault)或者寫入位置錯(cuò)誤。
第5行代碼是正確的,可以更改指針變量本身的指向;第6行代碼是錯(cuò)誤的,不能修改字符串中的字符。
到底使用字符數(shù)組還是字符串常量
在編程過程中如果只涉及到對字符串的讀取,那么字符數(shù)組和字符串常量都能夠滿足要求;如果有寫入(修改)操作,那么只能使用字符數(shù)組,不能使用字符串常量。
獲取用戶輸入的字符串就是一個(gè)典型的寫入操作,只能使用字符數(shù)組,不能使用字符串常量,請看下面的代碼:
#include <stdio.h>
int main(){char str[30];gets(str);printf("%s\n", str);return 0;
}
PS總結(jié):最后我們來總結(jié)一下,C語言有兩種表示字符串的方法,一種是字符數(shù)組,另一種是字符串常量,它們在內(nèi)存中的存儲(chǔ)位置不同,使得字符數(shù)組可以讀取和修改,而字符串常量只能讀取不能修改。
總結(jié)
以上是生活随笔為你收集整理的C指针5:字符串指针(指向字符串的指针)的全部內(nèi)容,希望文章能夠幫你解決所遇到的問題。
- 上一篇: C指针4:数组指针(指向数组的指针)
- 下一篇: C指针6:指针变量作为函数参数