转自:http://blog.sina.com.cn/s/blog_4c78b35f010008hi.html

笔者在开发某软件过程中遇到这样一个问题,前级模块传给我二进制数据,输入参数为 char* buffer 和 int length,buffer是数据的首地址,length表示这批数据的长度。数据的特点是:长度不定,类型不定,由第一个字节(buffer[0])标识该数据的类型,共有256(2的8次方)种可能性。我的任务是必须对每一种可能出现的数据类型都要作处理,并且我的模块包含若干个函数,在每个函数里面都要作类似处理。若按通常做法,会写出如下代码:

 void MyFunction(char* buffer, int length)
{
__int8 nStreamType = buffer[]
switch(nStreamType)
{
case :
function1();
break;
case :
......
case :
function255();
break;
}
}

如果按照这种方法写下去,那么在我的每一个函数里面,都必须作如此多的判断,写出的代码肯定很长,并且每一次处理,都要作许多次判断之后才能找到正确的处理函数,代码的执行效率也不高。针对上述问题,我想到了用函数指针数组方法解决这个问题。

函数指针的概念,在潭浩强先生的C语言程序设计这本经典的教程中提及过,在大多数情况下我们使用不到,也忽略了它的存在,函数名实际上也是一种指针,指向函数的入口地址,但它以不同于普通的如int*、double*指针,看下面的例子来理解函数指针的概念:

  int funtion(int x, int y)
void main(void)
{
int(*fun)(int x, int y)
int a = , b = ;
function( a, b)
fun = function;
(*fun)(a, b);
.......
}

语句1 定义了一个函数function,其输入为两个整数,返回也为一个整形数(输入参数和返回值可为其它任意数据类型);

语句3 定义了一个函数指针,与int* 或double* 定义指针不同的是,函数指针的定义必须同时指出输入参数,表明这是一个函数指针,并且*fun也必须用一对括号起来;

语句6 将函数指针赋值为function,前提条件是*fun和function的输入参数和返回值必须保持一到。

语句5 直接调用函数functio()。

语句7 是调用函数指针,二者等效。

当然从上述例子看不出函数指针的优点,目的主要是想引出函数指针数组的概念。我们从上面例子可以得知,既然函数名可以通过函数指针加以保存,那么也一定能定义一个数组保存若干个函数名,这就是函数指针数组。正确使用函数指针数组的前提条件是,这若干个这样,我工作中所面临的问题可以解决如下:

首先定义256个处理函数(及其实现)。

void function0(void);

......

void funtion255(void);

其次定义函数指针数组,并给数组赋值。

void(*fun[256])(void);

fun[0] = function0;

.....

fun[255] = function();

最后MyFunction()函数可以修改如下:

  void MyFunction(char* buffer, int length)
{
__int8 nSteamType = buffer[];
(*fun[nStreamType])();
}

只要2行代码,就完成了256条语句要做的事,减少了编写代码时工作量,将nStreamType作为数组下标,从代码执行效率上来说,也比case语句高。假如多个函数中均要作如此处理,函数指针数组更能体现出它的优势。

函数指针与typedef

关于C++中函数指针的使用(包含对typedef用法的讨论)

(一)简单的函数指针的应用。

  //形式1:返回类型 (*函数名)(参数表)
char (*pFun)(int);
char glFun(int a){ return; }
void main()
{
pFun = glFun;
(*pFun)();
}

第一行定义了一个指针变量pFun。首先我们根据前面提到的;“形式1”认识到它是一个指向某种函数的指针,这种函数参数是一个int型,返回值是char类型。只有一句我们还无法使用这个指针,因为我们还未对它进行赋值。

第二行定义了一个函数glFun()。该函数正好是一个以int为参数返回char的函数。我们要从指针的层次上理解函数--函数的函数名实际上就是一个指针,函数名指向该函数的代码在内存中的首地址。

然后就是可爱的main()函数了,它的第一名您应该看得懂了--它将函数glfun的地址赋值给变量pFun。main()函数的第二句中“*pFun”显然是取pFun所指向的地址的内容,当然也就是取出了函数glFun()的内容,然后给定参数为2。

(二)使用typedef更直观更方便

  //形式2:typedef 返回类型(*新类型)(参数)
typedef char(*PTRFUN)(int);
PTRFUN pFun;
char glFun(int a){ return;}
void main()
{
pFun = glFun;
(*pFun)();
}

typedef的功能是定义新的类型。

第一行就是定义了一种PTRFUN的类型,并定义这种类型为指向某种函数的指针,这种函数以以一个int为参数并返回char类型。后面就可以像使用int,char一样使用PTRFUN了。

第二行的代码使使用这个新类型定义了变量pFun,此时就可以像使用形式1一样使用这个变量了。

(三)在C++类中使用函数指针:

   //形式3:typedef 返回类型(类名::*新类型)(参数表)
class CA
{
public:
char lcFun(int a){ return; }
};
CA ca;
typedef char(CA::*PTRFUN)(int);
PTRFUN pFun;
void main()
{
pFun = CA::lcFun;
ca.(*pFun)();
}

在这里,指针的定义与使用都加上了“类限制”或“对象”,用来指明指针指向的函数是哪个类的,这里的类对象也可以是使用new得到。比如:

  CA* pca = new CA;
pca->(*pFun)();
delete pca;

而且这个类对象指针可以是类内部成员变量,你甚至可以使用this指针。比如:

  void CA::lcFun2()
{
(this->*m_pFun)();
}

一句话,使用类成员函数指针必须有“->*”或“.*”的调用。

转:函数指针数组的妙用(I)的更多相关文章

  1. C++基础——函数指针 函数指针数组

    ==================================声明================================== 本文版权归作者所有. 本文原创,转载必须在正文中显要地注明 ...

  2. typedef 函数指针 数组 std::function

    1.整型指针 typedef int* PINT;或typedef int *PINT; 2.结构体 typedef struct { double data;}DATA,  *PDATA;  //D ...

  3. C#委托与C语言函数指针及函数指针数组

    C#委托与C语言函数指针及函数指针数组 在使用C#时总会为委托而感到疑惑,但现在总新温习了一遍C语言后,才真正理解的委托. 其实委托就类似于C/C++里的函数指针,在函数传参时传递的是函数指针,在调用 ...

  4. C/C++ 一段代码区分数组指针|指针数组|函数指针|函数指针数组

    #include<stdio.h> #include<stdlib.h> #include<windows.h> /* 举列子说明什么是函数指针 */ //以一个加 ...

  5. C/C++ 不带参数的回调函数 与 带参数的回调函数 函数指针数组 例子

    先来不带参数的回调函数例子 #include <iostream> #include <windows.h> void printFunc() { std::cout<& ...

  6. C 函数指针数组

    名字有点绕口,其实更应该翻译为指针函数数组. 记录下对Head-First C这一节的理解,几乎每天班车上都会咪两眼,几乎每次都是看不懂,敲一敲的时候才有些明白. 通俗点讲,这功能解决的是,具有同种签 ...

  7. C语言的函数指针数组(好绕啊~看完这篇估计就通关了)

    转自https://www.cnblogs.com/chr-wonder/p/5168858.html int *(*p(int))[3] 今天有人问这个是啥?我一看直接就懵逼了…… 下面做一些简单的 ...

  8. C++ code:函数指针数组

    函数指针作为一种数据类型,当然可以作为数组的元素类型.例如,要实现用菜单来驱动函数调用的程序框架,则用函数指针数组来实现就比较容易维护. #include<iostream> using ...

  9. c语言.函数指针数组

    函数指针: 一个指向函数的指针.一般用函数名表示. 函数指针数组:元素为函数指针的数组.转移表.c语言中函数不可以定义为数组,只能通过定义函数指针来操作. #include<stdio.h> ...

随机推荐

  1. Java并发(十一):Condition条件

    先做总结: 1.为什么使用Condition条件? synchronized配合Object的wait().notify()系列方法可以实现等待/通知模式. Lock提供了条件Condition,对线 ...

  2. Android 按钮长按下去重复执行某个动作,放开后停止执行动作

    Android开发中,常遇到一种需求,即按钮长按下去重复执行某个动作,放开后停止执行动作.网上找了许多代码,都没有适合的,于是自己动手写了一个. 基本思路是:首先设置一个标识变量,用于标识是否处于按下 ...

  3. 微信小程序背景音频播放分享功能

    如果正常背景音频播放的话,只能跳转到自己对应的微信小程序,无法分享朋友圈,我们需要设置分享朋友圈,需要调用一个API 音频背景播放 注意:背景播放在锁屏后播放只支持IOS端,安卓端虽然可以播放,但是锁 ...

  4. Java常量定义需要注意事项及static作用(复习)

    在任何开发语言中,都需要定义常量.在Java开发语言平台中也不例外.不过在Java常量定义的时候,跟其他语言有所不同.其有自己的特色.在这篇文章中,主要针对Java语言中定义常量的注意事项进行解析,帮 ...

  5. Raspberry pi,一个好玩的派:第八季 Raspbmc(下)

    上一季安装好Raspbmc就等着这一季好好玩耍呢. 我们要在这一季中完毕例如以下任务:调整分辨率.连接wifi并在无线路由器中设置固定IP.手机遥控Raspbmc.改变语言为中文.远程訪问Raspbe ...

  6. [Asp.net]DropDownList改变默认选中项的两种方式

    引言 其实是不想总结这方面的内容,发现太简单了,可是在这上面也栽了跟头.所以还是记录一下吧,算是提醒自己,不要太看不起太基础的东西,有这种心理,是会载大跟头的. 一个例子 这里模拟一下最常用的一个例子 ...

  7. 【GISER&&Painter】Chapter02:WebGL中的模型视图变换

    上一节我们提到了如何在一张画布上画一个简单几何图形,通过创建画布,获取WebGLRendering上下文,创建一个简单的着色器,然后将一些顶点数据绑定到gl的Buffer中,最后通过绑定buffer数 ...

  8. poj3126--Prime Path(广搜)

    Prime Path Time Limit: 1000MS   Memory Limit: 65536K Total Submissions: 11751   Accepted: 6673 Descr ...

  9. OpenShift 如何获取bearer Token以便进行各种API调用

    Openshift 需要通过bearer token的方式和API进行调用,比如基于Postman就可以了解到,输入bearer token后 1.如何获取Bearer Token 但Bearer T ...

  10. 9.线程通信wait、notify

    线程之间通信 1.线程是操作系统的独立的个体,但这些个体如果不经过特殊处理就不能成为一个整体. 2.使用wait.notify,方法实现线程通信(2个方法都是需要object方法) 3.wait(释放 ...