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