回 帖 发 新 帖 刷新版面

主题:【C++提高】成员函数指针在开发中的实际应用


来源:[url=http://dozb.blogchina.com]http://dozb.blogchina.com[/url]
作者:dozb

前几天我的《成员函数指针》一文,主要介绍了类成员函数指针的使用方法,大家看了后认为意义不大,那是因为你可能还没有真正在开发中遇到需要用成员函数指针的情况。这篇文章就作为《成员函数指针》一文补充,举例说明在什么情况下使用成员函数指针来简化开发。

设想我们要实现一个命令解释器类,就是这个类包括了若干个命令,通常我们的实现方法如下:

class CmdProcess
{
public:
     CmdProcess();
    int execCmd(char *cmd,int argc,char*argv[]);
    void getCmdHelp(char *cmd,char* cmdhelp);
private:
    int cmd_login(int argc,char*argv[]);
    int cmd_logout(int argc,char*argv[]);
    int cmd_putfile(int argc,char*argv[]);
    int cmd_getfile(int argc,char*argv[]);
    int cmd_listfile(int argc,char*argv[]);
    int cmd_delfile(int argc,char*argv[]);
};

//根据传入的命令名字和参数执行命令
int CmdProcess::execCmd(char *cmd,int argc,char*argv[])
{
    if(strcmp(cmd,"login")==0)
     return   cmd_login(argc,argv);
    else  if(strcmp(cmd,"logout")==0)
      return cmd_logout(argc,argv);
    else if(strcmp(cmd,"putfile")==0)
      return cmd_putfile(argc,argv);
    else if(strcmp(cmd,"getfile")==0)
     return  cmd_getfile(argc,argv);
    else if(strcmp(cmd,"listfile")==0)
     return  cmd_listfile(argc,argv);
    else if(strcmp(cmd,"delfile")==0)
     return  cmd_delfile(argc,argv);
     return -1;
}

//根据传入的命令名字获取命令的帮助信息
void CmdProcess::getCmdHelp(char *cmd,char* cmdhelp)
{
    if(strcmp(cmd,"login")==0)
       ...
    else  if(strcmp(cmd,"logout")==0)
       ...
    else if(strcmp(cmd,"putfile")==0)
       ...
    else if(strcmp(cmd,"getfile")==0)
       ...
    else if(strcmp(cmd,"listfile")==0)
       ...
    else if(strcmp(cmd,"delfile")==0)
       ...
}

从上面可以看出,命令函数只是函数名不同,而参数和返回值相同,但是每次都要遍历所有的命令,烦不烦?更要命的是要是新增加一个命令,就需要修改所有的函数!假如你忘了修改一个地方,嘿嘿,后果是什么你也猜到了吧。

下面是用函数指针来实现相同的功能,可能看起来复杂一些,但要是遵循这个规则,使用起来还是比较简单的。

class CmdProcess
{
public:
     CmdProcess();
    int execCmd(char *cmd,int argc,char*argv[]);
    void getCmdHelp(char *cmd,char* cmdhelp);
private:
    typedef int (CmdProcess::* CMDFUN)(int argc,char*argv[]);//命令函数格式定义
    #define _CMDCOUNT 6
    struct _CMDINFO
    {
    char szCmdName[10];//命令名称
    char szCmdHelp[100];//命令帮助信息
    CMDFUN cmdFun;    //命令函数
    _CMDINFO(char* szCmdName,char* szCmdHelp,CMDFUN cmdFun)
    {
            strcpy(this->szCmdName,szCmdName);
        strcpy(this->szCmdHelp,szCmdHelp);
        this->cmdFun = cmdFun;
    };
    };
    _CMDINFO cmdInfoList[_CMDCOUNT];
    _CMDINFO * getCmdInfo(char *cmd);
private:
    int cmd_login(int argc,char*argv[]);
    int cmd_logout(int argc,char*argv[]);
    int cmd_putfile(int argc,char*argv[]);
    int cmd_getfile(int argc,char*argv[]);
    int cmd_listfile(int argc,char*argv[]);
    int cmd_delfile(int argc,char*argv[]);
};

//初始化成员函数指针的值
CmdProcess::CmdProcess()
{
    int i=0;
    cmdInfoList[i++] = _CMDINFO("login","login ....",cmd_login);
    cmdInfoList[i++] = _CMDINFO("logout","logout ....",cmd_logout);
    cmdInfoList[i++] = _CMDINFO("putfile","putfile ....",cmd_putfile);
    cmdInfoList[i++] = _CMDINFO("getfile","getfile ....",cmd_getfile);
    cmdInfoList[i++] = _CMDINFO("listfile","listfile ....",cmd_listfile);
    cmdInfoList[i++] = _CMDINFO("delfile","delfile ....",cmd_delfile);
}

//根据命令字找命令
_CMDINFO * CmdProcess::getCmdInfo(char *cmd)
{
   for(int i=0;i<_CMDCOUNT;i++)
   {
    if(strcmp(cmd,cmdInfoList[i].szCmdName)==0)
       return &cmdInfoList[i];
   }
   return NULL;
}

//根据传入的命令字和参数执行命令
int CmdProcess::execCmd(char *cmd,int argc,char*argv[])
{
   _CMDINFO * pCmdInfo = getCmdInfo(cmd);
   if(pCmdInfo)
      return (this->*(pCmdInfo->cmdFun))(argc,argv);//注意如何使用成员函数指针
   return -1;
}

//根据传入的命令字获取命令帮助信息
void CmdProcess::getCmdHelp(char *cmd,char* cmdhelp)
{
   _CMDINFO * pCmdInfo = getCmdInfo(cmd);
   if(pCmdInfo)
      strcpy(cmdhelp,pCmdInfo->szCmdHelp);

}

这种方式,当新增命令时,需要修改命令个数,和把相应命令添加到cmdInfoList就可以了。

回复列表 (共5个回复)

沙发

up

板凳

强,顶

3 楼


极好,我是在类外面定义的指针,也就是说,指针不是该类的成员,极别扭,不用还不行,因为要反复选别哪一个函数需要调用,所以感到很难受,看到你的程序很好,不过还没有完全看懂,但还是谢谢,另外以一个问题:如下
class MyClass
{
public:
    MyClass(char theChar,.....);
    .....
    long double z(long double x,long double y);
    long double fx1(long double x,long double y);
    long double fx2(long double x,long double y);
};
以上函数中有两个函数fx1、fx2,如果我要在类中定义一个指向fx1或fx2的指针,根据theChar的内容(如,'a'或'b')来确定对象使用哪一个方法(fx1、fx2),fx1、fx2为两种计算导数的方法,如果我将指针定义成fx,则程序就会变得简单自然,如
char c='a';
MyClass theClass(c,...);
对空间某点的求导数计算方法就为:
theClass.(指针fx)(long double x,long double y);//该行肯定不对
如能定义成以上形式该多好,可惜我还没有成功
能帮一下老哥吗?44了,学得确实有点难

4 楼

实际区别不大,效率上还不如if .. else

就像你这个例子中,只凭命令字来分发调用,几乎是最理想的情况了。
一般还有对上下文,当前状态等等的判断,才能决定调用某个处理函数。
这里查表,还不如if .. else。

简单举个例子,windows的消息处理,除mfc以外的类库中,基本都选择了
switch .. case/if else,而不是查表。

5 楼

3楼大哥,不用搞啥函数指针(当然为了练习你也可以尝试),很简单写个成员函数封装一下你的fx1,fx2就可以了啊
long double fx(long double x,long double y)
{
    if(theChar=='c')
        return fx1(x,y);
    else
        return fx2(x,y);
}

然后你就可以private掉你的fx1和fx2了

我来回复

您尚未登录,请登录后再回复。点此登录或注册