數組的地址
- int arr[5] 數組名是低一維元素的地址arr[0]的地址。而數組的地址是&arr。兩者有很大的區別
- 示例:
結果:int main() { int a[2][3] = {0}; printf("%p\n", a); printf("%p\n", a + 1); printf("%p\n", &a[0][0]); printf("%p\n", &a[0][0] + 1); printf("%p\n", &a[0]); printf("%p\n", &a[0] + 1); printf("%p\n", &a); printf("%p\n", &a + 1); }
0028FF28 0028FF34 0028FF28 0028FF2C 0028FF28 0028FF34 0028FF28 0028FF40
- 我們發現 a的值是與a[0]的值相等。可以把數組名理解一個指針,其值就是低一維首元素的地址。
- 我們看到+1的步長,a+1中間隔了12個字節,也就是3個int相當于第一維的長度。而&a+1的步長是整個數組的長度
指針數組
- int *a[3] 。為什么這里是指針數組。[]的優先級高于* ,所以這是一個數組,而*修飾數組,所以是指針數組,數組的元素是整型的指針。
- 示例:
結果:int main() { char *str[3]; str[0] = "abc"; str[1] = "efg"; str[2] = "hij"; for (int i = 0; i < 3; ++i) { printf("%s\n", str[i]); } }
abc efg hij
數組指針
- int (*a)[3]。同樣的方式,首先括號的優先級最高,所以*a是指針,而[]修飾*a ,所以是數組指針,一個指向3個元素的一維數組指針。
- 示例:
結果:typedef int arr[3]; int main() { arr b = {1, 2, 3}; int (*a)[3] = &b; arr *c = a; for (int i = 0; i < 3; ++i) { printf("%d\n", (*a)[i]); } }
1 2 3
- 解析:
- 這里使用typedef。我們自定義了一個數據類型,為數組數據類型。起數據類型為三個整型元素的數組。
- 定義數組指針也有兩種方式,一個是使用我們上面自定義的數組數據類型,一個是直接定義。
注意
- 在判斷變量是到底是數組還是指針或者使用自定義的數據結構。我們可以從操作符的優先級入手,看變量的具體是什么類型同時什么作為修飾。