C++ 函数指针赋值的基本规则
C++中函数指针的赋值核心是将目标函数的地址赋予指针变量,赋值的前提是指针的类型和函数的类型完全匹配,包括返回值类型、参数个数和参数类型都要一致,否则会出现编译错误。

普通非成员函数指针的赋值
普通函数的函数指针赋值相对简单,函数名本身就代表函数的地址,可以直接将函数名赋值给同类型的函数指针,也可以通过取地址符&获取函数地址后赋值,两种方式效果一致。
首先定义函数指针类型和对应的普通函数:
// 定义函数指针类型,指向返回值为int,参数为两个int的函数
typedef int (*FuncPtr)(int, int);
// 定义两个匹配的普通函数
int add(int a, int b) {
return a + b;
}
int sub(int a, int b) {
return a - b;
}
接下来进行赋值操作:
#include <iostream>
using namespace std;
int main() {
FuncPtr ptr1;
// 方式1:直接赋值函数名
ptr1 = add;
cout << "ptr1指向add的结果:" << ptr1(3, 2) << endl;
FuncPtr ptr2;
// 方式2:使用取地址符赋值
ptr2 = ⊂
cout << "ptr2指向sub的结果:" << ptr2(3, 2) << endl;
return 0;
}
类成员函数指针的赋值
类的非静态成员函数指针赋值需要注意,成员函数属于类而不属于对象,赋值时需要使用类作用域限定符,同时调用时也需要绑定对应的对象实例。
定义类和对应的成员函数指针类型:
class Calculator {
public:
int mul(int a, int b) {
return a * b;
}
int div(int a, int b) {
return b != 0 ? a / b : 0;
}
};
// 定义成员函数指针类型,需要指定所属类
typedef int (Calculator::*MemberFuncPtr)(int, int);
成员函数指针的赋值和调用示例:
#include <iostream>
using namespace std;
int main() {
MemberFuncPtr mptr;
// 赋值成员函数,必须加类作用域限定符
mptr = &Calculator::mul;
Calculator cal;
// 调用成员函数指针,需要绑定对象实例
cout << "成员函数指针调用结果:" << (cal.*mptr)(4, 5) << endl;
mptr = &Calculator::div;
cout << "成员函数指针调用结果:" << (cal.*mptr)(10, 2) << endl;
return 0;
}
赋值时的常见注意事项
- 函数指针和函数的返回值、参数列表必须完全匹配,比如返回值为
void的指针不能赋值给返回值为int的指针类型。 - 普通函数指针不能直接赋值类的非静态成员函数,反之亦然,两者的类型不兼容。
- 静态成员函数属于类但不依赖于对象实例,其指针赋值方式和普通函数指针一致,不需要类作用域的特殊处理。
- 赋值后可以通过指针直接调用函数,也可以先判断指针是否为空再调用,避免空指针调用导致的程序崩溃。
函数指针赋值的应用场景
函数指针赋值后最常见的应用是实现回调函数,比如在排序算法中传入不同的比较函数指针,就能实现升序或降序排序。另外也可以在状态机中根据不同状态赋值不同的处理函数指针,简化分支逻辑。
#include <iostream>
#include <algorithm>
#include <vector>
using namespace std;
// 升序比较函数
bool ascend(int a, int b) {
return a < b;
}
// 降序比较函数
bool descend(int a, int b) {
return a > b;
}
int main() {
vector<int> nums = {3, 1, 4, 1, 5, 9};
// 赋值升序比较函数指针给sort的第三个参数
sort(nums.begin(), nums.end(), ascend);
cout << "升序排序结果:";
for (int num : nums) {
cout << num << " ";
}
cout << endl;
// 赋值降序比较函数指针
sort(nums.begin(), nums.end(), descend);
cout << "降序排序结果:";
for (int num : nums) {
cout << num << " ";
}
cout << endl;
return 0;
}