c語言數組詳解
這篇文章主要介紹了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語言數組的理解,如果錯誤的地方,謝謝指正,輕噴哈。