声明函数指针实现回调

转载 2011年01月15日 10:39:00

程序员常常需要实现回调。本文将讨论函数指针的基本原则并说明如何使用函数指针实现回调。注意这里针对的是普通的函数,不包括完全依赖于不同语法和语义规则的类成员函数(类成员指针将在另文中讨论)。

 

声明函数指针

   回调函数是一个程序员不能显式调用的函数;通过将回调函数的地址传给调用者从而实现调用。要实现回调,必须首先定义函数指针。尽管定义的语法有点不可思议,但如果你熟悉函数声明的一般方法,便会发现函数指针的声明与函数声明非常类似。请看下面的例子:

void f();// 函数原型

上面的语句声明了一个函数,没有输入参数并返回void。那么函数指针的声明方法如下:

void (*) ();

   让我们来分析一下,左边圆括弧中的星号是函数指针声明的关键。另外两个元素是函数的返回类型(void)和由边圆括弧中的入口参数(本例中参数是空)。注意本例中还没有创建指针变量-只是声明了变量类型。目前可以用这个变量类型来创建类型定义名及用sizeof表达式获得函数指针的大小:

// 获得函数指针的大小
unsigned psize = sizeof (void (*) ());

// 为函数指针声明类型定义
typedef void (*pfv) ();

pfv是一个函数指针,它指向的函数没有输入参数,返回类行为void。使用这个类型定义名可以隐藏复杂的函数指针语法。

指针变量应该有一个变量名:

void (*p) (); //p是指向某函数的指针

   p是指向某函数的指针,该函数无输入参数,返回值的类型为void。左边圆括弧里星号后的就是指针变量名。有了指针变量便可以赋值,值的内容是署名匹配的函数名和返回类型。例如:

void func()
{
/* do something */
}
p = func;

p的赋值可以不同,但一定要是函数的地址,并且署名和返回类型相同。

传递回调函数的地址给调用者

   现在可以将p传递给另一个函数(调用者)- caller(),它将调用p指向的函数,而此函数名是未知的:

void caller(void(*ptr)())
{
ptr(); /* 调用ptr指向的函数 */
}
void func();
int main()
{
p = func;
caller(p); /* 传递函数地址到调用者 */
}

   如果赋了不同的值给p(不同函数地址),那么调用者将调用不同地址的函数。赋值可以发生在运行时,这样使你能实现动态绑定。下面看我写的demo程序
#include "stdafx.h"
#include "iostream.h"
typedef void (*pCallBack)();//指向返回值为void,无参的函数

void call(pCallBack pFun)//参数的类型是函数的指针
{
 pFun();
}

void test()
{
 cout<<"callback fun test"<<endl;
}
void main()
{  

 call(test);//print callback fun test

}
调用规范

   到目前为止,我们只讨论了函数指针及回调而没有去注意ANSI C/C++的编译器规范。许多编译器有几种调用规范。如在Visual C++中,可以在函数类型前加_cdecl,_stdcall或者_pascal来表示其调用规范(默认为_cdecl)。C++ Builder也支持_fastcall调用规范。调用规范影响编译器产生的给定函数名,参数传递的顺序(从右到左或从左到右),堆栈清理责任(调用者或者被调用者)以及参数传递机制(堆栈,CPU寄存器等)。

   将调用规范看成是函数类型的一部分是很重要的;不能用不兼容的调用规范将地址赋值给函数指针。例如:

// 被调用函数是以int为参数,以int为返回值
__stdcall int callee(int);

// 调用函数以函数指针为参数
void caller( __cdecl int(*ptr)(int));

// 在p中企图存储被调用函数地址的非法操作
__cdecl int(*p)(int) = callee; // 出错


   指针p和callee()的类型不兼容,因为它们有不同的调用规范。因此不能将被调用者的地址赋值给指针p,尽管两者有相同的返回值和参数列。

 

本文来自CSDN博客,转载请标明出处:http://blog.csdn.net/sendy888/archive/2007/07/31/1719445.aspx

C++回调函数和this指针的实例

在做到界面开发的过程中,往往需要通过子窗口去操作父窗口的问题。方法很多,大致实现下回调函数和类静态对象的使用。 在类的非静态函数的形参列表里,有个隐含的this指针,但是静态成员函数没有this指针...
  • u010177010
  • u010177010
  • 2016年03月09日 17:26
  • 892

C/C++之回调函数---函数指针的作用

今天讨论下C/C++中的回调函数。      在理解“回调函数”之前,首先讨论下函数指针的概念。 函数指针 (1)概念:指针是一个变量,是用来指向内存地址的。一个程序运行时,所有和运行相关的物件都是需...
  • kkk0526
  • kkk0526
  • 2013年12月04日 17:24
  • 5369

【C++基础之八】函数指针和回调函数

C++很多类库都喜欢用回调函数,MFC中的定时器,消息机制,hook机制等待,包括现在在研究的cocos2d-x中也有很多的回调函数。 1.回调函数 什么是回调函数呢?回调函数其实就是一个通过函数指针...
  • jackyvincefu
  • jackyvincefu
  • 2013年09月16日 09:40
  • 27967

C语言学习笔记(21) 函数指针、回调函数及复杂指针阅读技巧

摘要:总结了函数类型的本质,回调函数的思想,以及负责指针阅读技巧右左法则。 一、函数类型     1.C语言中的函数也有自己的类型,就像数组一样。     2.C语言中国的函数的类型由函...
  • Deep_l_zh
  • Deep_l_zh
  • 2015年12月17日 22:20
  • 623

深入浅出剖析C语言函数指针与回调函数(一)

今天我们要搞明白的一个概念叫回调函数。什么是回调函数?百度的权威解释如下:回调函数就是一个通过函数指针调用的函数。如果你把函数的指针(地址)作为参数传递给另一个函数,当这个指针被用来调用其所指向的函数...
  • morixinguan
  • morixinguan
  • 2017年03月24日 00:19
  • 4466

回调函数定义

回调函数
  • user_920
  • user_920
  • 2015年06月30日 10:24
  • 1924

函数指针与回调函数详解

函数指针与回调函数详解1.什么是函数指针?函数指针就是指针。这个指针存放一个函数的地址,而函数的名称就该函数的入口,即地址。这类似于数组名就是数组的首地址。我们可以通过反汇编直观的查看到函数名和函数地...
  • men_wen
  • men_wen
  • 2016年09月29日 14:08
  • 2066

回调指针回调函数的使用(C语言)

回调函数,顾名思义,就是使用者自己定义一个函数,使用者自己实现这个函数的程序内容,然后把这个函数作为参数传入别人(或系统)的函数中,由别人(或系统)的函数在运行时来调用的函数。函数是你实现的,但由别人...
  • miao19920101
  • miao19920101
  • 2017年07月21日 15:59
  • 341

C/C++回调方式系列之一 函数指针和函数回调模式

一、函数指针 1. 函数的定义 return_type function_name(parameter list) { function_body }...
  • SweetTool
  • SweetTool
  • 2017年05月20日 15:45
  • 1392

C++中回调函数的一个简单例子?

回调函数应用实例: 1、定义一个Person类 (Person.h)文件: 注意:在这个类中指定了回调函数,回调函数的执行者,和回调函数指针 重要的是  回调函数和回调函数指针是怎么关联的? ...
  • yjhdxflqm
  • yjhdxflqm
  • 2016年01月08日 14:13
  • 1718
内容举报
返回顶部
收藏助手
不良信息举报
您举报文章: 声明函数指针实现回调
举报原因:
原因补充:

(最多只允许输入30个字)