首先是简单的一维数组
定义一个数组int arr[5]; arr是指向数组第一个元素的int类型指针,arr+1就是指针数组第二个元素的int类型的指针,*arr就是取这个指针里面对应的值,这个比较好理解
int arr[5] = { 1, 2, 3, 4, 5 };//定义数组printf("\n%x", *arr);//1 =arr[0]printf("\n%x", *(arr+1));//2 =arr[1]
定义一个二维数组int arr[2][3]
int arr[2][3] = { { 1, 2, 3 }, { 4, 5, 6 } };printf("\n%d", sizeof(*(&arr)));//24printf("\n%d", sizeof(*arr));//12
二维数组分为行和列,这是一个两行三列的数组,
第一个是&arr是取整个数组的指针,对应类型是int[2][3] ,所以它的对应值的大小是24(每个数字占四个字节)。
第二个 arr代表第一行的数组指针,这里可能需要大家动动脑子,在第一个例子一维数组int arr[3] 里面,arr默认就是第一个元素所对应的指针,类型是int,而指针自增每次的增加量=sizeof(你所用的数据类型)
所以arr+1的值会加4(因为一维数组所对应的元素的指针类型是int,4个字节),所以+1会找到数组的下个元素地址,并*地址求得指针地址对应的值,但是二维数组的里面的值不是int类型,而是int[3]
,所以arr是指向二维数组的第一行的指针,也就是对应里面的{1,2,3}的值,所以它的大小是12个字节
这个时候我们又运行了一段程序
int arr[2][3] = { { 1, 2, 3 }, { 4, 5, 6 } };printf("\n%x", arr);//printf("\n%x", *arr);//
发现两个的值都是这个数组的首地址,第一个是数组第一行的指针,所以直接打印会打印出第一行的首地址很正常,我们能理解
但是第二个不是应该打印第一行的值吗?
原来是因为使用整行数据没有实际的含义,编译器遇到这种情况都会转换为指向该行第 0 个元素的指针
就像一维数组的名字,在定义时或者和 sizeof、& 一起使用时才表示整个数组,出现在表达式中就会被转换为指向数组第 0 个元素的指针。
所以如果我们输入 **arr就会打印出该行的第0列的元素的值,也就是1