1、数组指针
数组指针首先是一个指针,这个指针存放着一个数组的首地址。
定义
int (*p1)[4];
p1是指向数组的指针,所以p1的数据类型是int [4],p1+1,p1会向后移动16个字节,p1就是一个行指针。
#include <stdio.h>
int main(void) {
int arr1[3][4] = {1,2,3,4,5,6,7,8,9,10,11,12};
int (*p1)[4] = arr1;
int arr2[4] = {1,5,3,9};
int (*p2)[4] = &arr2;
printf("%d\n",p1[1][2]);
printf("%d\n",*(*(p1+1)+1));
printf("%d\n",*(*(p2)+1));
printf("%p\n",arr1);
printf("%p\n",arr1+1);
printf("%p\n",arr2);
printf("%p\n",arr2+1);
return 0;
}
运行的结果是
7
6
5
00000016D5BFF960
00000016D5BFF970
00000016D5BFF950
00000016D5BFF954
当二维数组赋值给数组指针时,可以直接将二维数组的首地址赋值给p1,而一维数组赋值给数组指针时,要将一位数组的第一个元素的首地址赋值给p2
这是因为在C语言中,赋值符号“=”号两边的数据类型必须是相同的,p1和arr1都代表一个行指针,两者的操作是相同的,所以可以赋值。arr2是int类型的,arr2+1,arr2会向后移动4个字节,p2是一个行指针,所以把arr2的首地址赋值给p2,现在p2指向arr2的首地址
2、指针数组
指针数组首先是个数组,这个数组的每个元素都是指针
定义
int *p1[3];
p1中每个元素都是int *类型的
#include<stdio.h>
int main()
{
int arr1[3][4] = {1,2,3,4,5,6,7,8,9,10,11,12};
int *p1[3];
for (int i=0; i<3; i++) {
p1[i] = arr1[i];
}
printf("%d\n",*(*(p1+2)+1));
printf("%d\n",p1[2][1]);
return 0;
}
运行的结果是
10
10
3、数组指针与指针数组的区别
指针数组和数组指针形成的根本原因是运算符的优先级问题。在C语言中,()= [ ] > *。“[ ]”的优先级比“*”要高。指针先与“[ ]”结合,构成一个数组的定义,int *修饰的是数组的内容,即数组的每个元素。至于数组指针就更好理解了,在这里“()”比“[ ]”更靠近左,“*”号和指针变量构成一个指针的定义,int 修饰的是数组的内容,即数组的每个元素。
二者都可以用来表示二维指针。