函数指针
前言:
先看两个基础,函数指针和extern关键字,然后由一个具体的例子,具体使用下函数指针。
一、基础
函数指针:即指向函数的指针,本质还是一个指针。 函数指针的声明:返回值类型 ( * 指针变量名) ([形参列表]); 注意这里是声明不是定义,声明之后它就是一个类型了(与int,char,int *等级别等同,这点有点像结构体),然后就可以定义、使用了。 举例如下(下面这段小程序摘自百度百科):
int max(int x,int y){return (x>y? x:y);}
int main()
{
int (*ptr)(int, int);
int a, b, c;
ptr = max;
scanf("%d%d", &a, &b);
c = (*ptr)(a,b);
printf("a=%d, b=%d, max=%d", a, b, c);
return 0;
}
extern:extern可以置于变量或者函数前,以标示变量或者函数的定义在别的文件中,提示编译器遇到此变量和函数时在其他模块中寻找其定义。此外extern也可用来进行链接指定。也就是说extern有两个作用,第一个,当它与"C"一起连用时,如: extern "C" void fun(int a, int b);则告诉编译器在编译fun这个函数名时按着C的规则去翻译相应的函数名而不是C++的,C++的规则在翻译这个函数名时会把fun这个名字变得面目全非,可能是fun@aBc_int_int#%$也可能是别的,因为C++支持函数的重载。 第二,当extern不与"C"在一起修饰变量或函数时,如在头文件中: extern int g_Int; 它的作用就是声明函数或全局变量的作用范围的关键字,其声明的函数和变量可以在本模块活其他模块中使用,记住它是一个声明不是定义!也就是说B模块(编译单元)要是引用模块(编译单元)A中定义的全局变量或函数时,它只要包含A模块的头文件即可,在编译阶段,模块B虽然找不到该函数或变量,但它不会报错,它会在连接时从模块A生成的目标代码中找到此函数。
二、举例
这里说一下背景,假如我这里做一个平台,我一套代码要交付到多个产品,然后结合产品代码进行使用,那么我做平台肯定不能为每个产品做一套代码,但是有时候同样一个功能,各个产品之间会出现差异,这个时候指针函数就派上用场了,我给各个产品提供一个指针函数定义的变量,然后各个产品将自己的实现函数挂接在上面,这样就屏蔽了各个产品的差异,甚至有些产品可以不挂接,那么我就判断一下,如果没挂接,我就给一个默认的实现就ok了(这部分在下面代码中没体现).
代码如下:
平台代码:
#include<iostream>
#include"lib_main.h"
using namespace std;
funcs g_hook_func;
void hook_func_init()
{
g_hook_func.func1=NULL;
g_hook_func.func2=NULL;
}
int main()
{
char name[10];
int result = 0 ;
memset(name,0,sizeof(name));
hook_func_init();
hook_func();//钩子挂接函数,多线程情况应该在产品侧挂接
if(g_hook_func.func1 != NULL)
{
if(0 == g_hook_func.func1(name))
{
cout<<"err";
return -1;
}
}
if(g_hook_func.func2 != NULL)
{
result = g_hook_func.func2(1,2);
}
cout<<name<<" "<<result<<endl;
return 0;
}
平台头文件:
#ifndef _LIB_MAIN_H_
#define _LIB_MAIN_H_
typedef struct func
{
int (*func1)(char * str);
int (*func2)(int a,int b);
}funcs;
extern void hook_func();
#endif
产品1代码:
#include"wlan.h"
#include<iostream>
using namespace std;
static int getname(char * str)
{
if(NULL == str)
{
return 0;
}
//入参大小由调用者保证不越界
str[0]='w';
str[1]='l';
str[2]='a';
str[3]='n';
str[4]=' ';
return 1;
}
static int add(int a, int b)
{
return (a+b+3);
}
void hook_func()
{
g_hook_func.func1 = getname;
g_hook_func.func2 = add;
}
产品1头文件代码:
#ifndef _WLAN_H_
#define _WLAN_H_
typedef struct func
{
int (*func1)(char * str);
int (*func2)(int a,int b);
}funcs;
extern funcs g_hook_func;
#endif
产品2代码:
#include"ar.h"
#include<iostream>
using namespace std;
static int getname(char * str)
{
if(NULL == str)
{
return 0;
}
//入参大小由调用者保证不越界
str[0]='a';
str[1]='r';
str[2]=' ';
return 1;
}
static int add(int a, int b)
{
return (a+b+1);
}
void hook_func()
{
g_hook_func.func1 = getname;
g_hook_func.func2 = add;
}
产品2头文件:
#ifndef _AR_H_
#define _AR_H_
typedef struct func
{
int (*func1)(char * str);
int (*func2)(int a,int b);
}funcs;
extern funcs g_hook_func;
#endif
产品3代码:
#include"fw.h"
#include<iostream>
using namespace std;
static int getname(char * str)
{
if(NULL == str)
{
return 0;
}
//入参大小由调用者保证不越界
str[0]='f';
str[1]='w';
str[2]=' ';
return 1;
}
static int add(int a, int b)
{
return (a+b+2);
}
void hook_func()
{
g_hook_func.func1 = getname;
g_hook_func.func2 = add;
}
产品3头文件:
#ifndef _FW_H_
#define _FW_H_
typedef struct func
{
int (*func1)(char * str);
int (*func2)(int a,int b);
}funcs;
extern funcs g_hook_func;
#endif
说明:
1、上面的所有代码不能同时运行,想一下也应该知道,应该是一套lib和一套产品放在一个工程下运行。
2、多线程条件下挂接钩子的函数hook_func应该在产品侧挂接,这样即使没有挂接,在lib侧也没有影响。
3、平台和产品侧的结构都要进行声明,且要一致typedef struct func { int (*func1)(char * str); int (*func2)(int a,int b); }funcs;
注意这里是声明,不是定义,所以不会分配内存,声明只是表示我这里现在有了这种类型(就像是说我这里有一个int一样)
- 深入理解计算机系统读书笔记之第一章:漫游
- 【实战】工控网络协议模糊测试:用peach对modbus协议进行模糊测试
- 【Django错误】OSError: raw write() returned invalid length 14 (should have been between 0 and 7)
- P2234 [HNOI2002]营业额统计
- Python 中格式化字符串 % 和 format 两种方法之间的区别
- [实战]如何在Kali Linux中进行WIFI钓鱼?
- jQuery
- P3369 【模板】普通平衡树(Treap/SBT)
- Django之ORM基础
- P3381 【模板】最小费用最大流
- Django基本命令
- docker 基本命令
- python3的一些简单高级用法(未完待续)三元运算生成列表字典等if or 简化lambda 函数+filter 函数
- Django基础
- JavaScript 教程
- JavaScript 编辑工具
- JavaScript 与HTML
- JavaScript 与Java
- JavaScript 数据结构
- JavaScript 基本数据类型
- JavaScript 特殊数据类型
- JavaScript 运算符
- JavaScript typeof 运算符
- JavaScript 表达式
- JavaScript 类型转换
- JavaScript 基本语法
- JavaScript 注释
- Javascript 基本处理流程
- Javascript 选择结构
- Javascript if 语句
- Javascript if 语句的嵌套
- Javascript switch 语句
- Javascript 循环结构
- Javascript 循环结构实例
- Javascript 跳转语句
- Javascript 控制语句总结
- Javascript 函数介绍
- Javascript 函数的定义
- Javascript 函数调用
- Javascript 几种特殊的函数
- JavaScript 内置函数简介
- Javascript eval() 函数
- Javascript isFinite() 函数
- Javascript isNaN() 函数
- parseInt() 与 parseFloat()
- escape() 与 unescape()
- Javascript 字符串介绍
- Javascript length属性
- javascript 字符串函数
- Javascript 日期对象简介
- Javascript 日期对象用途
- Date 对象属性和方法
- Javascript 数组是什么
- Javascript 创建数组
- Javascript 数组赋值与取值
- Javascript 数组属性和方法
- JSP分页显示(前端处理)
- 拓扑排序-HDU2647 Reward
- hadoop2.7.3源码解析之HA架构分析
- hadoop源码解析之hdfs内部结构分析
- 浙大版《C语言程序设计(第3版)》题目集 习题10-5 递归计算Ackermenn函数
- 浙大版《C语言程序设计(第3版)》题目集 习题10-6 递归求Fabonacci数列
- hadoop2.7.3源码解析之hdfs删除文件全流程分析h
- 二维树状数组-POJ 2155 Matrix
- 浙大版《C语言程序设计(第3版)》题目集 习题10-7 十进制转换二进制
- flink开发过程中遇到的问题集锦
- FLINK实战-使用CEP进行网站监控报警和报警恢复
- 浙大版《C语言程序设计(第3版)》题目集 习题10-8 递归实现顺序输出整数
- 浅谈DAO设计模式(示例)
- 浙大版《C语言程序设计(第3版)》题目集 习题11-1 输出月份英文名
- flink教程-flink 1.11 使用sql将流式数据写入hive