我对OOP和C++还是陌生的,我正在尝试学习良好的习惯。

我正在重组我继承的c++项目。我最大的困扰是他们如何实现命令 shell 解释器/解析器。它们基本上具有最大的if / elseif列表,每个可能的响应功能都在一个文件中处理。它可以在一个文件中快速删除多达数千行代码。这不是很有效(对命令进行二进制搜索会更快很多),并且要导航文件以添加新命令/响应,甚至跟踪现有响应的行为都非常艰巨。

我想做的是两件事:

1)将所有这数百种功能都放在一个文件中,然后将它们分组为几个外部文件,并根据我决定的一些高级抽象概念进行分组。

2)有一个很大的表/数组,其中包含一个字符串命令,并且是适当的函数指针响应。我将对此进行排序,并在适当时进行二进制搜索。我希望将此表初始化在一个地方,因此,如果有需要添加的新命令,可以将其追加到表中。一处一桌地将命令映射到响应

所以我想知道在实现这些文件和表的情况下,您的c++ / oop宗师会提出什么建议。就像我说的那样,我还很新,所以我想知道所有这些新的外部文件是否应该只包含静态函数和变量(只是一个“命名空间”文件),还是应该使它们成为成熟的类。类的概念听起来不错,但是我不知道在将这些对象的方法指针用于编译时初始化表时是否会存在某种实例化排序问题。在这种情况下,单例类(class)是否有意义?

最佳答案

一个非常简单的解决方案是让一个类按名称注册和调用您的函数:

class FunctionRegistry {
public:
    typedef std::function< void() > Function;

    void registerFunction( const std::string &name, Function f );
    void call( const std::string &name );

    static FunctionRegistry &instance();
private:
    FunctionRegistry();
    FunctionRegistry( const FunctionRegistry & );
    ...
};

// cpp file
FunctionRegistry &FunctionRegistry::instance()
{
     static FunctionRegistry theRegistry;
     return theRegistry;
}

如果您想将某些内容传递给该函数并使用std::bind来分配签名不完全匹配的函数,则可以修改函数签名和方法调用。在该类中,您将具有一个映射或哈希映射<std::string, Function>。您可能需要添加诊断,例如何时已注册具有特定名称的功能以及何时调用未注册但详细信息的功能。

现在,您可能需要一个助手:
class FunctionRegistrator {
public:
    FunctionRegistrator( const std::string &name, FunctionRegistry::Function f )
    {
        FunctionRegistry::instance().registerFunction( name, f );
    }
};

现在,您可以根据需要在cpp文件中进行任何方式的功能分组:
namespace {
    FunctionRegistrator regfunc1( "func1", std::bind( func1 ) );
    ...
}

该cpp文件中的每个功能。
如果您不能使用c++11,则可能需要使用boost::bind和boost::function。

07-24 09:44
查看更多