C語言char s[]和char* s的區別
C語言指針可以代替數組使用
1、數組本質
數組是多個元素的集合,在內存中分佈在地址連續的單元中,因此可以通過其下標訪問數組的不同數組。
例如:
下面展示一些
char s[3] = "0x1"; printf("s2字符串:\n"); printf("%c\n", s[0]); printf("%c\n", s[1]); printf("%c\n", s[2]);
2、指針
指針也是一種變量,隻不過它的內存單元中保存的是一種標識其他位置的地址,而地址也是整數,在32位平臺下,就是32位,4個字節(bytes)。
指針的指向
指針的指向是指 指針變量所保存的其他的地址單元中 所存放的數據類型。
例如:
int *ptr;//ptr指針保存的地址所在內存單元中的數據類型是整型 float *p;//這個p指針指向的內存地址存放的元素類型就是浮點型
而不管指向的數據類型是哪種,其實對於指針本身的值永遠是整型,因為它保存的地址就是整數
3、字符數組
字符數組首先是數組,數組中的元素都是字符。
char s[10];//定義瞭一個含有10個元素的數組,元素類型為字符 %c
C語言中定義一個變量時,可以初始化,如下:
char s[10] = {"hello world"};
當編譯器遇到上面的定義和賦值時,會將 hello world 和 \0 依次逐個填入連續數組內存中。
C語言中是沒有真正的字符串類型,可以通過字符數組表示字符串,並且字符數組的元素地址也是連續的。C語言中規定數組代表數組所在內存位置的首地址,即 str字符串是等於str[0]的,str = &str[0];
對於printf(“%s”,str); 為什麼用首地址就可以輸出字符串?
因為對於C語言中字符串常量的本質就是一個地址,例如:
char *s ; s = "Hello";
這裡把一個字符串賦值給一個字符串指針變量,起始就是C語言中編譯器會給字符串常量分配地址,如果”Hello”, 存儲在內存中的 0x3000 0x3001 0x3002 0x3003 0x3004 0x3005 ,可以看出真正的意義就是
s = “Hello” = 0x3000
因此我們可以把Hello看做是字符串,而編譯器會把他看做是地址 0x3000 ,即字符串常量的本質就是代表它的第一個字符的地址。
char *s; s = "Hello"; printf("%p\n",s);//%p代表表示按十六進制輸出數據,如果輸出數據不夠8位數,則左邊補0
輸出00796BD0,也就是”Hello”的首地址。
對於字符數組:
char str[10] = “Hello”;
也就是說str = &str[0]
C語言中操作字符串是通過它在內存中的存儲單元的首地址進行的,這是字符串的終極本質。。。
4、char * 與 char a[ ]
對於char *s 與 char a[ ] :
a代表的是字符串的首地址,而s代表的這個指針保存字符串的首地址(第一個字符的地址),即可以看做: s =a ,即可以將數組名賦值給指針表示地址,但是 不能這樣賦值 a = s ,即:不能將指針賦值給數組名
因為數組名是一個常量,是不可以修改的
可以通過如下方式訪問數組元素:
char a[] = "Hello"; char*s = a; int i; for(i = 0;i < strlen(a);i++) { printf("%c",s[i]); //printf("%c",*s++);也可以 }
字符指針可以用 間接操作符 * 取其內容,也可以用數組的下標形式 [ ],數組名也可以用 *操作,因為它本身表示一個地址 。。
比如 printf(“%c”,*a); 將會打印出 ‘H’,
char * s 與 char a[ ] 的本質區別
當定義char a[10]的時候,編譯器會給數組分配10個單元,每個單元的數據類型都是字符,而定義char *s時,s是一個指針變量,隻占4個字節,32位,用來保存一個地址。
sizeof(a) = 10,sizeof(s) = 4
a的長度是10,s的長度是4,因為s是一個int型
printf(“%p”,s)———-這個表示 s 的單元中所保存的地址。
printf(“%p”,&s);——–這個表示變量本身所在內存單元地址。。。。,不要搞混瞭。。
總結:char * s 隻是一個保存字符串首地址的指針變量,char a[]是許多連續的內存單元,單元中的元素是char型,char * 和 char a[]具有相同的效果,源於字符串的本質,即給一個字符串地址,便可以操作字符串,但char* 和 char a[]的本質屬性不一樣。
5、char ** 和char *a[]
char *a[]
[]的優先級高於 * ,所以先是a[],它是一個數組,數組中的元素是char類型,char元素是一個變量,保存地址。
對於char *a[ ] = {“Dog”,“Cat”,“Chicken”};
數組中元素是字符串,sizeof(a)是12,並不是4+4+8,是因為字符串常量的本質是地址,a數組中元素是char *指針,指針變量是整數類型,占4個字節,則3個元素就是12個字節。
char *a[] = { "Dog","Cat","Chicken" }; printf("%p %p %p \n", a[0], a[1], a[2]);
可以看到數組中的三個元素保存瞭三個內存地址,這三個地址代表瞭三個字符串的首地址,而不是字符串本身,且三個地址不是連續的,它是編譯器為”Dog”,“Cat”,“Chicken” 分配的內存空間的地址,因此沒有關聯。
char *a[] = { "Dog","Cat","Chicken" }; printf("數組元素單元本身的地址:%p %p %p \n", &a[0], &a[1], &a[2]);//數組元素單元本身的地址
可以看到三個元素單元所在的地址是連續的,每個地址相差四個字節。
char ** s
char ** 為二級指針,s保存一級指針char *的地址,
例如:
char *a [ ] = { "Dog","Cat","Chicken" }; char **s = a;//---有問題
數組a代表數組元素內存單元的首地址,,即a = &a[0] = 010FFD44,即 *s = 001D6BE0 = “Dog”,
這樣可以通過s訪問a中的數據
printf(“%s”,*s); printf("%s",a[0]); printf("%s",*a);
三個一樣,,但是不能把a = s,因為a是一個常量;
char **s = “hello world”; ——是錯誤的;
因為s的類型是char ** ,而 “hello world”的類型是char *;
雖然都是地址, 但是指向的類型不一樣,因此,不能這樣用。,從其本質來分析,“Hello world”,代表一個地址,比如0x000001,這個地址中的內容是 ‘H’,為char型,而s也保存一個地址,這個地址內容是char*,是一個指針類型。
char **s; *s = "hello world";
編譯沒有問題,但是 printf(“%s”,s),就會崩潰, printf (“%s”, s); 時,首先得有s 保存的地址,再在這個地址中找到 char * 的地址,即*s;
若s = 0x1000;
在0x1000所在的內存單元保存瞭“hello world”的地址0x000001,*s = 0x000001,這樣printf(“%s”,*s)會先找到0x1000,然後找到0x000001,如果直接使用char **s,令 * s = “hello world”,s變量中保存的是一個無效隨機不可用的值,不知道他指向哪裡,所以用char **s時,要給他分配一個內存地址。
char **s ; s = (char **) malloc(sizeof(char**)); *s = "hello world";
這樣s給分配瞭一個可用的地址,s = 0x1234,然後再地址0x1234所在的內存中的位置,保存瞭“hello world”的值。
下列程序中,定義瞭字符指針s,s中存放瞭字符串”message”的地址。
#include <stdio.h> void buf( char **s) { *s = "message"; } int main() { char *s ; buf(&s); printf("%s\n",s); }
即:二級指針保存的是一級指針的地址,它的類型就是指針變量,一級指針保存的是指向數據所在的內存單元的地址。
參考:
https://blog.csdn.net/daiyutage/article/details/8604720
到此這篇關於C語言char s[]和char* s的區別的文章就介紹到這瞭,更多相關C語言 char s[]和char* s內容請搜索WalkonNet以前的文章或繼續瀏覽下面的相關文章希望大傢以後多多支持WalkonNet!
推薦閱讀:
- C語言變長數組使用詳解
- C++中獲取字符串長度的函數sizeof()、strlen()、length()、size()詳解和區別(推薦)
- C程序中Ubuntu、stm32的內存分配問題
- C語言通過gets和gets_s分別實現讀取含空格的字符串
- C語言指針如何實現字符串逆序反轉