数组指针
数组指针的定义:
int (*p)[10];
//p先和*结合,说明p是一个指针变量,然后指针指向一个大小为10个整形的数组。所以p是一个指针,指向一个数组,叫指针数组。
//[]的优先级高于*号的,所以必须加上()来保证p和*先结合。
&数组名和数组名:
int arr[10];
arr是数组名,也就是首元素的地址。而 &arr 表示的是数组的地址 ,而不是首元素的地址,数组的地址加1,跳过的是整个数组的大小,所以 &arr+1 相对 &arr 的差值是40.
一个数组指针的使用:(打印二维数组)
#include<stdio.h>
viod print_arr1(int arr[3][5], int row, int col)
{
int i = 0;
int j = 0;
for(i=0; i<row; i++)
{
for(j=0; j<col; j++)
{
printf("%d ", arr[i][j]);
}
printf("\n");
}
}
viod print_arr2(int arr*[5], int row, int col)
{
int i = 0;
int j = 0;
for(i=0; i<row; i++)
{
for(j=0; j<col; j++)
{
printf("%d ", arr[i][j]);
}
printf("\n");
}
}
int main()
{
int arr[3][5] = {1,2,3,4,5,6,7,8,9,0};
print_arr1(arr,3,5);
print_arr2(arr,3,5);
//数组名是arr,标识的是首元素的地址
//但是二维数组的首元素是二维数组的第一行
//这里传递的arr,相当于第一行的地址,是一维数组的地址。
//可以使用数组指针来接收。
return 0;
}
指针数组:
是数组,用来存放指针的数组
如int * arr[5];
arr先和[]结合,说明是一个数组。然后数组里面的5元素是整形指针。
注意在二维数组传参时,函数的形参设计只能省略第一个[]的数字。
因为对一个二维数组可以不知道多少行,但是必须知道一行多少个元素。运算方便。
函数指针:
void test()
{
}
- void (*pfun1)();
函数指针,pfun1先和*结合,说明pfun1是指针,指针指向一个函数,指向的函数无参数,返回值类型是viod。
- void *pfun2();
返回值类型是void*类型的无参函数。
函数名表示函数的地址
解读经典的两个例子:
//代码1
(*(void)(*)())0)();
解析:
第一步: (void)(*)()是一个没有参数返回值类型是void的函数指针。
第二步: ((void)(*)())0 是将0强制类型转换为函数指针类型,0是一个地址,也就是说一个函数存在首地址为0的一块区域内。
第三步:(*(void)(*)())0) 对0地址进行解引用,期内容就是保存在首地址为0的一段区域内的函数。
第四步: (*(void)(*)())0)(), 就是函数调用。
//代码2
void(*signal(int, void(*)(int)))(int);
解析:
signal 是一个函数声明
signal函数的第一个参数是int,第二个参数是函数指针
signal函数的返回类型为函数指针
signal函数的第二个参数是函数指针,该指针指向的函数
参数是int,返回类型为void
signal函数的返回类型也是一个函数指针,该函指针指向的函数
参数是int,返回类型为void
函数指针数组
把函数的地址存到一个数组中,这个数组就叫做函数指针数组。
如: int (* a[10])();
指向函数指针数组的函数
是一个指针,指向一个数组,数组的元素都是函数指针。
回调函数
回调函数就是一个通过函数指针调用的函数
如果你把函数的指针作为参数传递给另一个函数,当这个指针被用来调用所指向的函数时,我们称其是回调函数。