什么是函数指针
如果在程序中定义了一个函数,那么在编译时系统就会为这个函数代码分配一段存储空间,这段存储空间的首地址称为这个函数的地址。而且函数名表示的就是这个地址。既然是地址我们就可以定义一个指针变量来存放,这个指针变量就叫作函数指针变量,简称函数指针。
那么这个指针变量怎么定义呢?虽然同样是指向一个地址,但指向函数的指针变量同我们之前讲的指向变量的指针变量的定义方式是不同的。例如:
int(*p)(int, int);
这个语句就定义了一个指向函数的指针变量 p。首先它是一个指针变量,所以要有一个“*”,即(*p);其次前面的 int 表示这个指针变量可以指向返回值类型为 int 型的函数;后面括号中的两个 int 表示这个指针变量可以指向有两个参数且都是 int 型的函数。所以合起来这个语句的意思就是:定义了一个指针变量 p,该指针变量可以指向返回值类型为 int 型,且有两个整型参数的函数。p 的类型为 int(*)(int,int)。大家需要记住这种类型,前面int表示返回值,中间括号括起来表示是一个指针,后面的括号表示参数类型列表。
所以函数指针的定义方式为:
函数返回值类型 (* 指针变量名) (函数参数列表);
“函数返回值类型”表示该指针变量可以指向具有什么返回值类型的函数;“函数参数列表”表示该指针变量可以指向具有什么参数列表的函数。这个参数列表中只需要写函数的参数类型即可。
我们看到,函数指针的定义就是将“函数声明”中的“函数名”改成“(*指针变量名)”。但是这里需要注意的是:“(*指针变量名)”两端的括号不能省略,括号改变了运算符的优先级。如果省略了括号,就不是定义函数指针而是一个函数声明了,即声明了一个返回值类型为指针型的函数。
那么怎么判断一个指针变量是指向变量的指针变量还是指向函数的指针变量呢?首先看变量名前面有没有“*”,如果有“*”说明是指针变量;其次看变量名的后面有没有带有形参类型的圆括号,如果有就是指向函数的指针变量,即函数指针,如果没有就是指向变量的指针变量。
最后需要注意的是,指向函数的指针变量没有 ++ 和 -- 运算。
如何用函数指针调用函数
给大家举一个例子:
# include <stdio.h>
int Max(int, int); //函数声明
int main(void)
{
int(*p)(int, int); //定义一个函数指针
int a, b, c;
p = Max; //把函数Max赋给指针变量p, 使p指向Max函数
printf("please enter a and b:");
scanf("%d%d", &a, &b);
c = (*p)(a, b); //通过函数指针调用Max函数
printf("a = %d\nb = %d\nmax = %d\n", a, b, c);
return 0;
}
int Max(int x, int y) //定义Max函数
{
int z;
if (x > y)
{
z = x;
}
else
{
z = y;
}
return z;
}
输出结果是:
please enter a and b:3 4
a = 3
b = 4
max = 4
我们再举一个例子:
#include <iostream>
#include <algorithm>
#include <math.h>
#include <vector>
#include <map>
#include <unordered_map>
#include <stack>
#include <set>
#include <string>
#include <functional>
using namespace std;
int getMax(int a, int b){
if (a >= b){
return a;
}
return b;
}
int getMin(int a, int b){
if (a <= b){
return a;
}
return b;
}
int mymap(string str, int a, int b){
map<string, int(*)(int, int)>mp;
mp["getMin"] = &getMin;
mp["getMax"] = &getMax;
return (*mp[str])(a, b);
}
int main(){
cout << mymap("getMax", 2, 6) << endl;
cout << mymap("getMin", 1, 7) << endl;
system("pause");
return 0;
}
那么怎么把类成员函数作为函数指针使用呢?
看看下面的例子:
#include <iostream>
#include <algorithm>
#include <math.h>
#include <vector>
#include <map>
#include <unordered_map>
#include <stack>
#include <set>
#include <string>
#include <functional>
using namespace std;
class pointertest{
public:
pointertest();
~pointertest(){};
public:
int func1(int a, int b);
int func2(int a, int b);
int func3(int a, int b);
int realfunc(string str, int a, int b);
unordered_map<string, int(pointertest::*)(int, int)> mp;
};
pointertest::pointertest(){
mp["func1"] = &pointertest::func1;
mp["func2"] = &pointertest::func2;
mp["func3"] = &pointertest::func3;
}
int pointertest::func1(int a, int b){
cout << "func1" << endl;
return a + b;
}
int pointertest::func2(int a, int b){
cout << "func2" << endl;
return a + b;
}
int pointertest::func3(int a, int b){
cout << "func3" << endl;
return a + b;
}
int pointertest::realfunc(string str, int a, int b){
//return (this->*(mp[str]))(a, b);
return (this->*(mp[str]))(a, b);
}
int getMax(int a, int b){
if (a >= b){
return a;
}
return b;
}
int getMin(int a, int b){
if (a <= b){
return a;
}
return b;
}
int mymap(string str, int a, int b){
map<string, int(*)(int, int)>mp;
mp["getMin"] = &getMin;
mp["getMax"] = &getMax;
return (*mp[str])(a, b);
}
int main(){
pointertest a;
int s = a.realfunc("func1", 3, 4);
cout << s << endl;
s = a.realfunc("func2", 3,5);
cout<<s<<endl;
system("pause");
return 0;
}
/*输出是:
func1
7
func2
8
*/