函数指针与回调函数

    函数有它的地址,程序运行起来了,程序里肯定有地方要放这个函数。我们知道可以用printf  %p来输出一个变量的地址,数组的地址,同样我们定义了一个函数后,用这个函数的名字做输出,就可以得到这个函数的地址。那么我们得到一个函数的地址,有什么用呢?

    我们可以想一下,在程序里我们定义一个变量i,然后定义一个指针p,让指针p指向i得到i的地址,然后通过*p我们就可以对i进行赋值等操作。那么,既然函数的名字可以得到函数地址,也就是说我们应该也可以用一个指针指向一个函数,得到函数的地址后,对函数做点事情,比如通过指针来调用这个函数?我们来试一下:

    程序编译通过了,但是给了一个warningwarning里说这个“函数指针的初始化是有矛盾的”。但是程序通过了编译,也就是说我们是可以用一个指针指向一个函数的,只是初始化时没有对应上类型,因为我们定义的指针pint*类型,而函数是void()类型。找对类型还不够,因为定义一个函数指针它还有特定的格式。

一、定义函数指针

定义函数指针的格式:例如:void *p)();

     void表示返回的类型,第二个括号是参数表。这样格式的意思是我们定义了一个(*p)函数指针。之后我们就可以对它做初始化,比如让它指向f函数:

    这样编译程序就没有了warning。那么定义了一个函数指针,我们让它指向了函数f,之后我们怎样调用指针里的内容?也就是调用指针所指的那个f函数?

二、调用函数指针

调用函数指针的格式:例如:(*p)();

我们来试一下这样的方式调用函数指针p所指的那个函数:

f函数里我们让它输出一句话,这样当我们的函数指针成功调用所指的函数后,函数里的那句话就会被输出,我们就知道函数调用成功了。

成功进入了f函数。

那么我们可以用函数指针来做些什么来使程序更方便灵活?

三、回调函数

    定义:把一个函数的指针作为参数传递到另一函数的参数表中,让这个函数的指针被调用它所指的那个函数时,这种行为就是回调函数。也就是说这个指针函数的调用方式不是直接调用,而是在特定的条件下由另一方调用。

我们先来看这样一个例子:

我们定义了一个f函数和一个g函数,两个函数里分别做输出。主函数里,第1617行定义两个函数指针一个指向f函数一个指向g函数。接下来第18行我们输一个值给i,然后做判断,如果i等于0就调用f函数,等于1就调用g函数。

可是这样有一个缺点是,如果以后我们想在程序里加个新的函数进去,那么就要修改写很多东西,比如在这个程序中,如果新添加一个h函数,当用户输入2时就调用h函数,那么就要在24行那里再加一个else if,这样改动很麻烦。

要解决这个问题,我们可以用指针数组的方式:

13行我们加了一个g函数,然后再主函数第21行中,我们定义了一个函数指针数组,(*fa[ ]),也就是这个数组里每一个元素都是一个指针。我们直接给它赋初值fgh。等程序22行输入了一个数后,23行判断,如果这个数是在有效范围内,就进入if做响应的函数调用。(sizeoffa/sizeoffa[0])得到该数组内元素的个数)。

这样做的好处是如果以后程序要做修改,比如新添加一个函数,调用它的话就在数组里加一个指针指向该函数即可。这种方式可以做成一个程序根据你输入的数去决定做什么事情,比如输入不同的数就调用不同的函数。

我们来看第二个例子,自定义一个函数用一个指针指向另一个函数:

f函数里给两个变量abreturn a+b的值,g函数同样给两个变量abreturn a-b的值。第13行我们定义一个h函数,函数要的参数是一个函数指针,并且这个函数指针的参数表里也要两个变量。h函数里我们输出这个函数指针所调用的那个函数,参数给23。在主函数中,第20行和21行分别用这个h函数调用f函数和g函数,得到的结果:

也就是说我们在程序中可以把一个函数当作一个值传入另一个函数里,在另一个函数里根据这个传入函数的值做事情。

第二个例子便是回调函数,回调函数的意义是,我们这个程序里的h函数所做的事情是固定的,就是输出函数指针所指的函数运行后得到得值。h函数所做的事情是根据外面传入的东西的不同而得到不同的结果。

  • 3
    点赞
  • 11
    收藏
    觉得还不错? 一键收藏
  • 4
    评论
函数指针回调函数是C语言中非常重要的概念,它们在实际编程中经常被用到。在本文中,我将对函数指针回调函数进行详细的解释,并通过生动的例子进行说明。 一、函数指针 函数指针是指向函数的指针变量。它可以像普通指针一样存储一个函数的地址,并且可以通过该指针调用该函数。在C语言中,函数名本身就是一个指向函数的指针,所以可以将函数名直接赋值给一个函数指针变量。 下面是一个简单的例子: ```c #include <stdio.h> int add(int a, int b) { return a + b; } int main() { int (*p)(int, int) = add; int result = p(1, 2); printf("%d\n", result); return 0; } ``` 在这个例子中,我们定义了一个函数add,它接受两个整数参数并返回它们的和。然后我们定义了一个函数指针变量p,它指向add函数。最后我们通过p指针调用add函数,并将结果打印出来。 函数指针的好处在于可以将函数作为参数传递给其他函数,这就是回调函数的应用场景。 二、回调函数 回调函数是指作为参数传递给另一个函数的函数。当该函数需要某些特定的操作时,它会调用传递进来的回调函数来完成这些操作。回调函数通常用于事件处理、异步编程等场合。 下面是一个例子: ```c #include <stdio.h> void print(int num) { printf("%d\n", num); } void traversal(int *arr, int size, void (*callback)(int)) { int i; for (i = 0; i < size; ++i) { callback(arr[i]); } } int main() { int arr[] = {1, 2, 3, 4, 5}; traversal(arr, 5, print); return 0; } ``` 在这个例子中,我们定义了一个print函数,它接受一个整数参数并将其打印出来。然后我们定义了一个traversal函数,它接受一个整数数组、数组长度以及一个回调函数作为参数。在traversal函数中,我们遍历整个数组,并对每个元素调用回调函数。最后我们在main函数中调用traversal函数,并将print函数作为回调函数传递进去。 这个例子展示了回调函数的一个常见应用场景:遍历数组或链表时,需要对每个元素执行相同的操作,但是具体的操作可以由调用者自己定义。 三、函数指针回调函数的结合应用 函数指针回调函数经常被结合使用。下面是一个例子: ```c #include <stdio.h> int add(int a, int b) { return a + b; } int sub(int a, int b) { return a - b; } void calculate(int a, int b, int (*callback)(int, int)) { int result = callback(a, b); printf("%d\n", result); } int main() { calculate(1, 2, add); calculate(3, 4, sub); return 0; } ``` 在这个例子中,我们定义了两个函数add和sub,它们分别实现了加法和减法运算。然后我们定义了一个calculate函数,它接受两个整数参数和一个回调函数作为参数。在calculate函数中,我们调用传递进来的回调函数,完成相应的运算。最后我们在main函数中调用calculate函数,分别传递add和sub函数作为回调函数。 这个例子展示了如何通过函数指针回调函数实现不同的运算操作,从而提高代码的灵活性和可复用性。 综上所述,函数指针回调函数是C语言中非常重要的概念,它们可以让我们通过代码来动态地指定函数的行为。在实际编程中,我们经常会使用到这两个概念,因此深入理解它们的原理和应用场景是非常有必要的。
评论 4
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值