一 數組名是什么
數組就是一段連續可用的內存。
比如聲明一個 int數組
int array[]={1,2,3};
array代表什么?有的資料說:數組名是指向數組首地址的常量指針。
下面我們可以驗證一下。
我都知道sizeof操作符可以返回一個對象或者類型所占的內存字節數。
如:
int i=1;
那么sizeof(i) 的結果就是4(64位機器下的部分編譯器是8)
那我們打印sizeof(array)
printf("%d\n",sizeof(array));
結果是:12。
但是我們都知道sizeof(指針變量)==4的。
所有我們得出:數組名不完全是指向數組首地址的常量指針。
為什么要用不完全,因為我們利用數組來訪問數組元素的時候。它又變得像一個常量指針。
比如
array[0]等效于*(array+0)
這時候 array就是一個指向數組首地址的常量指針,指針類型是指向數組元素類型的指針。這里就是 int*類型
我們可以這樣理解:
一個大學的名字叫array。
有人問你array是什么。你會告訴他array是大學,面積多少之類的。
但是有人問你去array怎么走,你會告訴他array的校門(首地址)在哪里。
結論:數組名其實代表的是一個內存區域,但是使用的時候變成了 指向數組首地址的常量指針。
但是這里還有一個小陷阱:
#include <stdio.h>
void foo(int a[])
{
printf("%d\n",sizeof(a));
}
int main(void)
{
int array[]={1,2,3};
foo(array);
return 0;
}
輸出的不是12,而是4。
出于效率的考慮,數組傳參是引用傳參而不是拷貝傳參。因為數組長度可能很大,拷貝一份的話太耗資源。
雖然我是這樣函數是這樣的
void foo(int a[])
{
printf("%d\n",sizeof(a));
}
編譯器的眼中是這樣
void foo(int *a)
{
printf("%d\n",sizeof(a));
}
所以sizeof(a)是sizeof(指針變量)肯定是4;
二 字符數組
首先我們看一個簡單的程序
# include <stdio.h>
int main(void)
{
char *str1="abc";
char str2[]="def";
printf("%s\n",str1+4);
return 0;
}
輸出的結果是 def。
我們要知道c語言中只要用到了 "xxxxx",系統都會自動的把雙引號的內容添加到字符常量區。
注意:printf("xxxx");不會把"xxxx"添加到字符常量區。
char *str1="abc"; //會把 abc\0 添加到字符常量區,并把首地址賦值給str指針變量。
char str2[]="def"; //會把 def\0 添加到字符常量區,并且函數棧中添加一個字符數組 內容也是 def\0,str2指向的是棧中的數組。
char str[]={'x','y','z'}; //只會在函數棧中添加數組
由于字符常量區是連續的,所以
printf("%s\n",str1+4);
可以打印出str2的值。
三 二維數組
int array[][3]={1,2,3,4,5,6};
前面我們已經說了,使用array訪問元素時,array就是一個指針類型為指向數組元素的指針類型,指向數組首地址的指針。
二維數組的元素就是數組,
這樣寫就更容易理解:
int array[][3]={{1,2,3},{4,5,6}};
所有可以 這樣認為 array是這樣認為的
int (*const array)[3];
當我訪問數組元素時候
array[x][y]在編譯器看來就是 *(*(array+x)+y)
*(array+x) 得到的是一個 第x行 類型為 “int[3]”(c語言沒有這樣的寫法) 的數組,
數組名在訪問元素的時候當做首地址指針來用,在這里 *(array+x)等同于數組名,
指針類型int *,指向的地址為 array+sizeof(int(*)[3])*x。
訪問這個數組的第y個元素的時候,就要用 *(*(array+x)+y)。
這些就是我對c語言數組的理解,如果錯誤的地方,謝謝指正,輕噴哈。