windows内核开发学习笔记十九:补充基本概念的学习——指针函数
今天学习内核编程,要补充一些基本的知识,以便后面学习时的概念的基本理解。今天介绍的是关于函数指针的概念。
定义
顾名思义,函数指针是关于函数的指针,函数指针虽然也是指针,但它的定义方式却和其他指针看上去很不一样,下面实例说明如何定义的:
/* 方法1 */
void (*p_func)(int, int, float) = NULL;
/* 方法2 */
typedef void (*tp_func)(int, int, float);
tp_func p_func = NULL;
这两种方式都是定义了一个指向返回值为 void
类型,参数为 (int, int, float)
的函数指针。第二种方法是为了让函数指针更容易理解,尤其是在复杂的环境下;而对于一般的函数指针,直接用第一种方法就行了。
函数指针的赋值
在定义完函数指针后,我们就需要给它赋值了我们有两种方式对函数指针进行赋值:
void (*p_func)(int, int, float) = NULL;
p_func = &func1;
p_func = func2;
两种方法都是合法的,对于第二种方法,编译器会隐式地将 func_2
由 void ()(int, int, float)
类型转换成 void (*)(int, int, float)
类型,因此,这两种方法都行。
使用函数指针调用函数
因为函数指针也是指针,因此可以使用常规的带 *
的方法来调用函数。和函数指针的赋值一样,我们也可以使用两种方法:
/* 方法1 */
int val1 = p_func(1,2,3.0);
/* 方法2 */
int val2 = (*p_func)(1,2,3.0);
方法1和我们平时直接调用函数是一样的,方法2则是用了 *
对函数指针取值,从而实现对函数的调用。
将函数指针作为参数传给函数
函数指针和普通指针一样,我们可以将它作为函数的参数传递给函数,下面我们看看如何实现函数指针的传参:
/* func3 将函数指针 p_func 作为其形参 */
void func3(int a, int b, float c, void (*p_func)(int, int, float))
{
(*p_func)(a, b, c);
}
/* func4 调用函数func3 */
void func4()
{
func3(1, 2, 3.0, func_1);
/* 或者 func3(1, 2, 3.0, &func_1); */
}
函数指针作为函数返回类型
有了上面的基础,要写出返回类型为函数指针的函数应该不难了,下面这个例子就是返回类型为函数指针的函数:
void (* func5(int, int, float ))(int, int)
{
...
}
在这里, func5
以 (int, int, float)
为参数,其返回类型为 void (*)(int, int)
。
函数指针数组
在开始讲解回调函数前,最后介绍一下函数指针数组。既然函数指针也是指针,那我们就可以用数组来存放函数指针。下面我们看一个函数指针数组的例子:
/* 方法1 */
void (*func_array_1[5])(int, int, float);
/* 方法2 */
typedef void (*p_func_array)(int, int, float);
p_func_array func_array_2[5];
上面两种方法都可以用来定义函数指针数组,它们定义了一个元素个数为5,类型是 void (*)(int, int, float)
的函数指针数组。