什么是函数指针

函数指针是指向函数的指针变量。也就是说,它是一个指针变量,而且该指针指向一个函数。

对于指针变量来说,它的值是它指向的变量的地址。举个例子:指针变量pi是指向一个整型变量i的指针,则变量i的地址 &i 就是指针变量pi的值。也就是说整型变量指针指向一个整型变量,而整型变量指针的值就是它所指的整型变量的地址。与其它类型指针变量一样,函数指针变量的值就是它指向的函数的地址。

那么什么是函数的地址呢?

我们首先来看函数调用是怎么回事。在程序运行时,一个函数占用一段连续的内存。当调用一个函数时,实际上是跳转到函数的入口地址,执行函数体的代码,完成后返回。

函数指针指向一个函数的入口地址,也就是函数存储空间的首地址。

在C语言中,数组名代表数组的首地址,同样函数名代表了函数的首地址,因此在赋值时,直接将函数指针指向函数名就行了。

函数指针的定义

一般,函数指针的定义格式为:

函数类型 (*指针变量名)(形参列表);

“函数类型”说明函数的返回类型,由于“()”的优先级高于“*”,所以指针变量名外的括号必不可少,后面的“形参列表”表示指针变量指向的函数所带的参数列表。

例如:对于函数int f(int a),我们定义一个指向该函数的函数指针fp,采用如下格式:

int (*fp)(int a);

函数指针的赋值

前面我们已经讲到,在C语言中,函数名代表了函数的首地址,因此在赋值时,直接将函数名赋值给函数指针就可以了。

例如:

int func(int x); //声明一个函数

int (*fp)(int x); //定义一个函数指针

fp = func; //将func函数的首地址赋值给指针fp

赋值时函数func不带括号,也不带参数,由于func代表函数的首地址,因此经过赋值以后,指针就指向函数func(x)的代码的首地址。

通过函数指针调用函数

与其它指针变量相类似,如果指针变量pi是指向某整型变量i的指针,则*pi等于它所指向的变量i;如果pf是指向某浮点型变量f的指针,则*pf就等价于它所指的变量f。同样地,fp是指向函数func(x)的指针,则*fp就代表它所指向的函数func。所以在执行了fp = func;之后,(*fp)和func代表同一函数。

由于函数指针指向存储区中的某个函数,因此可以通过函数指针调用相应的函数。

用函数指针调用函数由三步组成:

首先,定义函数指针变量。

例如:int (*fp)(int x);

然后,给函数指针变量赋值。

例如:fp = func; (func(x)必须要先有定义)

最后,用(*指针变量)(参数表);调用函数。

例如:(*fp)(x);(x必须先赋值)。

在这里,(*fp)(x);一般写成fp(x); fp(x)是标准C++的写法,(*fp)(x);是兼容C语言的标准写法。

总结以上知识,举例说明

到目前为止,相信您已经简单了解了函数指针,下面我们举一个简单的、完整的程序进行说明。

例1

#include <iostream>

using namespace std;

//定义一个函数add,两个输入参数为整型,返回值为整型

int add(int a,int b)

{

return a+b;

}

//定义一个函数指针变量,该变量所指函数的类型与int add(int,int)相对应。

int (*padd)(int,int);

int main()

{

int a1 = 1;

int b1 = 2;

//函数指针变量赋值,函数指针指向add函数

padd = add;

//通过函数指针变量调用函数,与直接调用add函数效果相同

int c1 = (*padd)(a1,b1);

//也可以写成int c1 = padd(a1,b1);

cout<<c1<<endl;

return 0;

}

编译并运行程序,返回两个数的和3。

函数指针的作用

现在,相信您已经明白了函数指针,但是,您可能会迷惑:函数指针到底有什么作用呢?

函数指针是C++一个重要的组成部分,它的最大好处就是实现函数的回调。

什么是回调函数

函数,我们大家都知道。函数体是为了完成某种功能的一段代码,一般情况下,我们在主程序中直接调用函数,主程序调用函数是为了完成这部分功能,被调用函数完成后立即返回主程序。

函数的调用,我们都知道,那么什么是回调函数,我们什么时候使用回调函数呢?

初次接触回调函数很难理解,我们先通过下面的图示简单地介绍回调函数。

图1

图1,我们很容易理解,它是我们正常调用函数的情况,程序中直接调用函数。

图2

图2的理解就有些费解,首先程序A调用函数1,而函数1本身又调用了程序A自己编写的函数2。函数2就是回调函数。

初次理解回调函数确实有些困难,为了让您弄懂它,我在这里再对图2进行一下解释:

在一般情况下,函数1是由系统完成的。图2 改为下图更容易理解:

图3

我们的程序执行了某项操作,会通知系统,调用系统的函数1执行相应功能,但系统还需调用函数2,函数2的内容是由我们来完成的。

比如,我们按下一个按钮,我们调用系统(也可以说通知系统)完成相应功能,系统首先完成按钮的重画,使其显示为按下状态,同时还要执行按钮被按下后的相应操作。按钮被按下后的操作内容是由用户来完成的,系统是不能确定的,用户可以根据自己的需要填写内容。

现在您明白了回调函数的好处了吧。它由程序开发人员完成其实现内容,而是由系统(或其它函数)调用,它使得我们编程更加方便灵活。

回调函数的实现

回调函数,是通过函数指针实现的。下面我们举一个最简单的通过函数指针实现回调函数的例子:

//下面的例程模拟单击按钮后的回调

//主函数调用的Function_After_Push_Button函数相当于按钮被按下后通知系统的函数

// OnClick函数是回调函数,相当于按钮被按下具体要执行什么操作,其内容由用户根据自己的需要来完成

#include <iostream>

using namespace std;

//定义回调函数OnClick

void OnClick ()

{

cout<<"The Button is Click!"<<endl;

}

//定义一个函数,该函数的输入参数是一个函数指针

void Function_After_Push_Button(void(*pf)())

{

pf();

}

int main()

{

//定义一个函数指针

void (*pf)();

//给函数指针赋值

pf = OnClick;

//实现回调

Function_After_Push_Button(pf);

return 0;

}

上例是一个非常简单的回调函数,但并不能完全体现出回调函数的好处。该例程旨在演示如何使用函数指针实现回调,在实际编程过程中,最常用的还是使用系统规定好的回调函数。

VC++的函数指针和回调函数 及友元函数的更多相关文章

  1. C语言之函数指针、回调函数的使用

    一.背景 首先看下如下代码,这个定义是放在头文件的,在程序中tCdrvCallbackFkt也定义了另一个变量,而且括号后面还跟定义了几个变量,不理解这个定义. typedef void (PUBLI ...

  2. C语言函数指针和回调函数

    彻底搞定C指针-函数名与函数指针 函数名&函数名取地址 函数指针 通常我们可以将指针指向某类型的变量,称为类型指针(如,整型指针).若将一个指针指向函数,则称为函数指针. 函数名的意义 函数名 ...

  3. C:函数指针、回调函数

    函数指针 是一个指针,指向函数的指针,指针存放的都是地址,所以函数指针存放的是函数的地址.数组名就是数组的首地址,函数名就是函数的首地址.与数组类似. 代码demo int (*p) (int ,in ...

  4. 利用C语言中的函数指针实现c++中的虚函数

    C语言中的函数指针 #include<stdio.h> int fun1(int a) { return a*a; } int fun2(int a) { return a*a*a; } ...

  5. C++成员函数指针错误用法警示(成员函数指针与高性能的C++委托,三篇),附好多评论

    今天做一个成绩管理系统的并发引擎,用Qt做的,仿照QtConcurrent搞了个模板基类.这里为了隐藏细节,隔离变化,把并发的东西全部包含在模板基类中.子类只需注册需要并发执行的入口函数即可在单独线程 ...

  6. 声明函数指针、回调函数、函数对象------c++程序设计基础、编程抽象与算法策略

    声明函数指针 #include<iostream> using namespace std; double a(double aa) { return aa; } int main() { ...

  7. 深入浅出剖析C语言函数指针与回调函数(一)【转】

    本文转载自:http://blog.csdn.net/morixinguan/article/details/65494239 关于静态库和动态库的使用和制作方法. http://blog.csdn. ...

  8. C 函数指针、回调函数

    参考链接:https://www.runoob.com/cprogramming/c-fun-pointer-callback.html 函数指针 函数指针就是执行函数的指针,他可以像正常函数一样去调 ...

  9. C基础知识(6):指针--函数指针与回调涵数

    函数指针 函数指针是指向函数的指针变量. 通常我们说的指针变量是指向一个整型.字符型或数组等变量,而函数指针是指向函数. 函数指针可以像一般函数一样,用于调用函数.传递参数. #include < ...

随机推荐

  1. 用efibootmgr管理UEFI启动项,添加丢失的启动项

    UEFI用来替代传统BIOS引导操作系统,学会修改UEFI启动项也变得十分重要,UEFI全称为:“统一的可扩展固件接口”(Unified Extensible Firmware Interface), ...

  2. Bootstrap概览

    摘选自runnoob.com,bootstrap4所有api概览: 一.安装: 弹性盒子(flexbox) Bootstrap 3 与 Bootstrap 4 最大的区别就是 Bootstrap 4 ...

  3. paste---合并文件的列。

    Linux paste命令用于合并文件的列. paste指令会把每个文件以列对列的方式,一列列地加以合并. 语法 paste [-s][-d <间隔字符>][--help][--versi ...

  4. 彻底解决Linux索引节点(inode)占用率高的告警

    今天邮箱里发现有一封某服务器inode使用率发生告警的邮件 登录到服务器上df -i查看,发现/路径下91%,磁盘使用率却不高,猜测可能是某个目录下的小文件过多,进而造成inode占用率过高,但不清楚 ...

  5. 【面试加分项】java自己定义注解之申明注解

    之前的博客http://blog.csdn.net/u010590685/article/details/47029447介绍了java的注解的基本知识今天我们学习怎样使用自己定义注解. 首先我们要声 ...

  6. HTTP服务器状态码定义

    HTTP服务器状态代码定义 1.1 消息1xx(Informational 1xx) 该类状态代码用于表示临时回应.临时回应由状态行(Status-Line)及可选标题组成, 由空行终止.HTTP/1 ...

  7. System.GC.Collect();//垃圾回收,回收没有正常关闭的http连接

    System.GC.Collect();//垃圾回收,回收没有正常关闭的http连接

  8. Netty系列之Netty编解码框架分析

    1. 背景 1.1. 编解码技术 通常我们也习惯将编码(Encode)称为序列化(serialization),它将对象序列化为字节数组,用于网络传输.数据持久化或者其它用途. 反之,解码(Decod ...

  9. userdel---删除用户及相关文件

    userdel命令   userdel命令用于删除给定的用户,以及与用户相关的文件.若不加选项,则仅删除用户帐号,而不删除相关文件. 语法 userdel(选项)(参数) 选项 -f:强制删除用户,即 ...

  10. 求s=a+aa+aaa+aaaa+aa...a的值,其中a是一个数字。 例如2+22+222+2222+22222(此时共有5个数相加),几个数相加有键盘控制。

    代码: package com.liron.p1; import java.io.IOException; import java.util.Scanner; /** * 求s=a+aa+aaa+aa ...