前面写了指针数组和数组指针,而这个话题也是异曲同工之妙。
同样的只看中文名称会糊糊涂涂,不明不白,难以理解记忆。而这两个英文名也十分类似:
pointer function(指针函数)和 function pointer(函数指针),后来我看到指针函数的另一种表达方式:function return pointer,问题得以较好的解决
不管如何我们先看开始吧:
- 指针函数:function return pointer;即:返回值为指针的函数。
- 函数指针:function pointer;即:指向函数的指针。
下面看详细用法:
指针函数:返回值为指针的函数
static int sum = 0;
/*******************************
* 声明形式:
* ret* func(args, ...);
*
* ret* :返回值类型
*
*******************************/
// 加
int * add(int n)
{
int *p = ∑
sum += n;
return p;
}
// 减
int * reduce(int n)
{
int *p = ∑
sum -= n;
return p;
}
int main(int argc, char *argv[])
{
QApplication app(argc, argv);
int *a = add(5);
// 打印数据:指针地址,指针指向的地址,指针指向的地址的值
qDebug() << &a << " : " << a << " : " << *a;
int *b = reduce(2);
// 打印数据:指针地址,指针指向的地址,指针指向的地址的值
qDebug() << &b << " : " << b << " : " << *b;
return app.exec();
}
运行结果:
0x34fd00 : 0x7ff6a65c5408 : 5
0x34fd18 : 0x7ff6a65c5408 : 3
这个两个指针函数都是返回全局变量sum的地址。指针函数在实际编程中常用到,用于返回摸个对象的指针,是一个必须掌握的点。
对于指针的释放控制,我们一般遵循谁创建谁释放,但当指针函数出现时,我们极有可能出现野指针的情况:返回出去的指针还在用,里面的对象已经被释放了。对于这种情况我们推出了智能指针的解决方案,后面再去说。
函数指针:指向函数的指针
static int sum = 0;
// 加
int add(int n)
{
sum += n;
return sum;
}
// 减
int reduce(int n)
{
sum -= n;
return sum;
}
int main(int argc, char *argv[])
{
QApplication app(argc, argv);
/*******************************
* 声明形式:
* ret (*p)(args, ...);
*
* 声明一个返回类型是 ret ,参数是(args, ...) 的函数指针
*
*******************************/
int (*p)(int); // 函数指针的定义
p = add; // 函数指针的赋值
int a = p(5); // 函数指针的使用
// 打印数据:指针地址,返回值
qDebug() << &p << ":" << a;
p = reduce;
a = p(3);
// 打印数据:指针地址,返回值
qDebug() << &p << ":" << a;
return app.exec();
}
运行结果:
0xb1f570 : 5
0xb1f570 : 2
函数指针也是一个常用的知识点。当我们希望把一个功能交给外部实现,而我们是要拿到对应的值时。或者在某个过程中我们想把某个值留个外部或者第三方操作时。我们都可以使用函数指针的方式去实现。其中最经典的事例:回调函数。
回调函数
回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数时,我们就说这是回调函数。回调函数不是由该函数的实现方直接调用,而是在特定的事件或条件发生时由另外的一方调用的,用于对该事件或条件进行响应。(来自百度百科)
- 提供函数实现的一方在初始化的时候,将回调函数的函数指针注册给调用者;
- 当特定的事件或条件发生的时候,调用者使用函数指针调用回调函数对事件进行处理。
static int sum = 0;
// 加
int add(int n)
{
sum += n;
return sum;
}
// 减
int reduce(int n)
{
sum -= n;
return sum;
}
int callback(int n, int (*p)(int))
{
return p(n);
}
int main(int argc, char *argv[])
{
QApplication app(argc, argv);
qDebug() << callback(5, add);
qDebug() << callback(3, reduce);
return app.exec();
}
运行结果:
5
2
理解:
add和reduce是回调函数;callback是调用者。