函数指针与回调函数

关于静态库和动态库的使用和制作方法。

http://blog.csdn.net/morixinguan/article/details/52451612

今天我们要搞明白的一个概念叫回调函数。

什么是回调函数?

百度的权威解释如下:

回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应

那么我们可以来看一个例子:

[cpp]  view plain  copy
 print ?
  1. #include <stdio.h>  
  2. void print();  
  3. int main(void)  
  4. {  
  5.     void (*fuc)();   
  6.     fuc = print ;   
  7.     fuc();    
  8. }   
  9. void print()  
  10. {  
  11.     printf("hello world!\n");  
  12. }  

从这个例子可以看到,我们首先定义了一个函数指针fuc ,这个函数指针的返回值为void型,然后我们给函数指针赋值,赋值为print,也就是print函数的首地址,此时fuc获得了print的地址,fuc的地址等于print的地址,所以最终调用fuc();也就相当于调用了print();

么我写的这个例子明显和百度解释的不符合啊?定义是如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数,确实,有所不同,但道理是一样的,我们接下来再来看一个例子。

[cpp]  view plain  copy
 print ?
  1. #include <stdio.h>  
  2.   
  3. int add_ret() ;  
  4.   
  5. int add(int a , int b , int (*add_value)())  
  6. {  
  7.     return (*add_value)(a,b);  
  8. }  
  9.   
  10. int main(void)  
  11. {  
  12.     int sum = add(3,4,add_ret);  
  13.     printf("sum:%d\n",sum);  
  14.     return 0 ;  
  15. }   
  16.   
  17. int add_ret(int a , int b)  
  18. {  
  19.     return a+b ;  
  20. }  

从这个例子里,我们看到:

这样子不就符合我们的定义了嘛?我们把函数的指针(地址),这里也就是add_ret,作为参数int add(int a , int b , int (*add_value)()) , 这里的参数就是int(*add_value)() , 这个名字可以随便取,但是要符合C语言的命名规范。当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。我们看到add函数内部,return (*add_value)(a,b) ; 这个(*add_value)(a,b)相当于对指针进行了简引用,我们在main函数中,传入具体要实现功能的函数,add_ret,这个函数很简单,就是实现两数相加并返回,这里刚刚好,简引用,相当于取出指针返回地址里的值,这个值就是return a+b,也就是我们传入a和b两数相加的结果。

         那么,回调函数究竟有什么作用呢?

说到这里,就有了用户和开发者之间的概念,比方说,刚刚说的add()这个函数,假设一下,用户是实现add这个函数,而开发者是实现add_value这个函数,用户做的工作不多,就是想要通过开发者提供实现两个数相加的这么一个接口,然后在函数中通过调用开发者实现的这个接口的返回值,然后来实现我们的功能。这个开发者角色就很多了,可以是自己公司的核心开发人物,也可以是别的工作的外包商的人物,这时候,他作为一个开发者的角色完完全全可以将add_value实现的add_ret这个函数封装起来并且加密,然后扔一个.so或者.a给用户,那么用户就看不到具体add_ret的实现内容,用户只需要开发者给他提供一个.h和.so即可,这样,作为开发者,他就将他实现的函数功能给保密了。

 接下来,我们用linux来演示下这个结果:

         我们在目录下创建三个文件,main.c,vendor.c,vendor.h

         Main.c是用户开发的

         Vendor.c和vendor.h是开发者实现的。

在main.c中,代码如下:

[cpp]  view plain  copy
 print ?
  1. #include <stdio.h>  
  2. #include "vendor.h"  
  3.   
  4. int add(int a , int b , int (*add_value)())  
  5. {  
  6.     return (*add_value)(a,b);  
  7. }  
  8.   
  9. int main(void)  
  10. {  
  11.     int sum = add(3,4,add_ret);  
  12.     printf("sum:%d\n",sum);  
  13.     return 0 ;  
  14. }   
vendor.c,代码如下:

[cpp]  view plain  copy
 print ?
  1. #include "vendor.h"  
  2. int add_ret(int a , int b)  
  3. {  
  4.     return a+b ;  
  5. }  
vendor.h,代码如下:

[cpp]  view plain  copy
 print ?
  1. #ifndef __VENDOR_H  
  2. #define __VENDOR_H  
  3.   
  4. int add_ret(int a, int b) ;  
  5.   
  6. #endif  

接下来,我们制作一个动态链接库,最终开发者把vendor.c的内容封起来,把vendor.h提供给用户使用。


在linux下制作动态链接库,将vendor.c和vendor.h打包成一个动态链接库

先明白以下几个命令是什么意思:

生成动态库:

gcc -shared -fPIC dvendor.c -o libvendor.so    

-shared : 生成动态库;

-fPIC  : 生成与位置无关代码;

-o               :指定生成的目标文件;

 

使用动态库:

gcc main.c -L . –lvendor -o main

-L : 指定库的路径(编译时); 不指定就使用默认路径(/usr/lib/lib)

-lvendor : 指定需要动态链接的库是谁;

代码运行时需要加载动态库:

./main 加载动态库 (默认加载路径:/usr/lib /lib ./ ...)

./main

我们将编译动态库生成的libvendor.so拷贝到/usr/lib后,现在就不需要vendor.c了,此时我们将vendor.c移除,也可以正常的编译并且执行main函数的结果,这就是回调函数的作用之一。


转自:http://blog.csdn.net/morixinguan



  • 0
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
函数指针回调函数是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语言中非常重要的概念,它们可以让我们通过代码来动态地指定函数的行为。在实际编程中,我们经常会使用到这两个概念,因此深入理解它们的原理和应用场景是非常有必要的。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

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

抵扣说明:

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

余额充值