1.指针数组:指针数组是数组,是存放指针的数组
1 int *arr1[10] //存放10个整形指针的数组
2 char *arr2[4]//存放4个字符指针的数组
3 char **arr3[5]//同样也是指针数组
2.数组指针:数组指针是指针
整形指针:int *pint;能够指向整形数据的指针
浮点型指针:float *pf;能够指向浮点型数据的指针
注意:int (*p)[10]//数组的地址存放到数组指针当中;(p先和*结合,说明p是一个指针变量,然后指针指向的是一
个大小为10的整形的数组。所以p是一个指针,指向一个数组,即数组指针。)
例: int arr[10];
int (*p)[10]=&arr;
int *arr[10];
int *(*p)[10]=&arr;
注意:以上的赋值都是合法的,[]的优先级要高于*号的,所以必须加上()来保证p先和*结合。
3.指针数组的使用:
int arr[10]={0};
arr;//表示数组首元素的地址
&arr;//表示数组的地址
printf("%p\n",arr);//首元素的地址
printf("%p\n",arr+1);数组中第二个元素的地址
printf("%p\n",&arr+1);数组最后一个元素之后的地址
(我们可以看出产生的结果是截然不同的,读者可以去验证,原因是数组的地址和首元素的地址是相同的,但意义却不同)
4.数组的地址通过数组指针存放。因此在传参的时候,如果将一个二维数组的数组名传给形参,我们知道,二维数组的数组
名即为该数组第一行的地址,也就是说该数组名为一个一维数组的地址,因此函数在接收实参传过来的值时应该用数组指针接收。
5.指针和数组的定义和声明
定义是不存在的时候要让他存在,而声明是不知道的要让他知道。
1 //test.c
2 //数组的定义
3 char arr[]="abcdef";
4 //指针的定义
5 char *p="abcdef";
1 //main.c
2 extern char arr[];
3 extern char *p;
4 int main()
5 {
6 printf("%s\n",arr);
7 printf("%s\n",p);
8 return 0;
9 }
读者可以自行试验,小编在这儿给出运行结果的截图:
a.定义为数组,声明为指针,运行结果如下图:
若要它正确输出,则可以以如下图的方式输出:
b.若定义为指针,声明为数组:
若要正确输出,则可以以下图两种方式输出:
注意:定义为数组,声明为指针时,程序崩溃,原因是将数组内存空间中的四个字节的内容(本例中为abcd的阿斯卡码值61626364)传给指针,指针将其视为一个地址,但该地址为非法地址,指针访问时出错,程序崩溃。定义为指针,声明为数组时,输出随机值,原因是将指针的地址以字符的形式输出,所以为随机值。
6.数组参数,指针参数
在写代码时难免要把数组或者指针传给函数,那函数的参数该如何设计呢?
a.一维数组传参:
1 #include
2 void test(int arr[])//一维数组传给一位数组,ok
3 {}
4 void test(int arr[10])//ok
5 {}
6 void test(int *arr)//ok
7 {}
8 void test2(int *arr[20])//指针数组传给指针数组
9 {}
10 void test2(int **arr)//ok
11 {}
12 int main()
13 {
14 int arr[10]={0};
15 int *arr2[20]={0};
16 test(arr);
17 test2(arr2);
18 return 0;
19 }
b.二维数组传参:
1 #include
2 void test(int arr[3][5])//ok
3 {}
4 void test(int arr[][])//错误,没有列
5 {}
6 void test(int arr[][5])//ok
7 {}
8 //总结:二维数组传参,函数形参的设计只能省略第一个[]的数字
9 //因为对于一个二维数组,可以不知道有多少行,但是必须知道一行多少元素。
10 void test(int *arr)//错误,因为arr是一个二维数组,数组名是第一行,也就是一维数组,不能把一位数组传给一级指针,应该传给数组指针
12 {}
13 void test(int *arr[5])//此处形参为指针数组,不是数组指针,错误
14 {}
15 void test(int (*arr)[5])//ok
16 {}
17 void test(int **arr)//错误,不能把一维数组传给二级指针
18 {}
19 int main()
20 {
21 int arr[3][5]={0};
22 test(arr);
23 return 0;
24 }
c.一级指针传参:
1 #include
2 void print(int *p,int sz)
3 {
4 int i=0;
5 for(i=0;i)
6 {
7 printf("%d\n",*(p+i));
8 }
9 }
10 int main()
11 {
12 int arr[10]={1,2,3,4,5,6,7,8,9};
13 int *p=arr;
14 int sz=sizeof(arr)/sizeof(arr[0]);
15 //一级指针p,传给函数
16 print(p,sz);
17 return 0;
18 }
d.二级指针传参:
#include
void test(int **ptr)
{
printf("num=%d\n",**ptr);
}
int main()
{
int n=0;
int *p=&n;
int **pp=&p;
test(pp);
test(&p);
return 0;
}
e.函数指针:
1 #include
2 void test()
3 {
4 printf("hehehe\n");
5 }
6 int main()
7 {
8 printf("%p\n",test);
9 printf("%p\n",&test);
10 return 0;
11 }
读者可以行验证,这里输出的是同一块空间的地址,说明函数名就是函数的地址。同样,可以将函数的地址存放到一个指针中,即函数指针,有了函数指针,便有了存放函数指针的数组,即函数指针数组,同理,有了函数指针数组,便有了函数指针的数组的指针。小编不在这里具体阐述。对于函数指针的用途,在很多地方都有应用,比如转移表,计算器,读者有兴趣的话可以阅读相关书籍。
指针数组 数组指针 函数指针 函数指针数组 指向函数指针数组的指针