全网整合营销服务商

电脑端+手机端+微信端=数据同步管理

免费咨询热线:400-708-3566

c++如何实现解释器模式 c++设计模式之Interpreter【实例】

解释器模式在C++中用于定义语言文法并构建解释器,适用于语法简单、执行频率低但需灵活扩展的场景,如布尔表达式计算;核心由AbstractExpression、TerminalExpression、NonterminalExpression和Context组成,通过递归下降解析器构建表达式树。

解释器模式(Interpreter Pattern)在 C++ 中用于定义语言的文法,并建立一个解释器来解释该语言中的句子。它适用于语法简单、执行频率不高、但需要灵活扩展语法规则的场景,比如简易表达式计算、配置脚本解析、规则引擎条件表达式等。

核心结构:抽象表达式 + 终结符/非终结符表达式

解释器模式的关键是将每个语法规则映射为一个类,所有表达式类型统一继承自抽象基类 Expression,并实现 interpret() 接口:

  • AbstractExpression:声明解释操作的接口(如 interpret(Context&)
  • TerminalExpression:对应终结符(如数字、变量名),不再分解,直接返回结果
  • NonterminalExpression:对应非终结符(如加、减、括号),内部持有子表达式,递归调用 interpret()
  • Context:封装解释器外部信息(如变量值表、全局状态)

实战示例:简易布尔表达式解释器

支持 ANDORNOT 和布尔字面量(True/False),输入字符串如 "True AND (NOT False)",输出 truefalse

关键代码片段(简化版):

class Context {
public:
    std::map variables;
};

class Expression {
public:
    virtual bool interpret(const Context& ctx) const = 0;
    virtual ~Expression() = default;
};

class BooleanLiteral : public Expression {
    bool value;
public:
    explicit BooleanLiteral(bool v) : value(v) {}
    bool interpret(const Context&) const override { return value; }
};

class VariableExpression : public Expression {
    std::string name;
public:
    explicit VariableExpression(const std::string& n) : name(n) {}
    bool interpret(const Context& ctx) const override {
        auto it = ctx.variables.find(name);
        return it != ctx.variables.end() ? it->second : false;
    }
};

class NotExpression : public Expression {
    std::unique_ptr expr;
public:
    explicit NotExpression(std::unique_ptr e) : expr(std::move(e)) {}
    bool interpret(const Context& ctx) const override {
        return !expr->interpret(ctx);
    }
};

class AndExpression : public Expression {
    std::unique_ptr left, right;
public:
    AndExpression(std::unique_ptr l, std::unique_ptr r)
        : left(std::move(l)), right(std::move(r)) {}
    bool interpret(const Context& ctx) const override {
        return left->interpret(ctx) && right->interpret(ctx);
    }
};

配合简易词法/语法分析(如递归下降解析器),即可将字符串构造成表达式树并执行。

如何构建表达式树?——手写递归下降解析器

不依赖第三方库时,可手写轻量解析器。例如对 "a AND NOT b",按优先级(括号 > NOT > AND/OR)分步解析:

  • 先识别标识符或字面量 → 构造 VariableExpressionBooleanLiteral
  • 遇到 NOT → 读取下一个表达式,包装成 NotExpression
  • 遇到 AND → 左侧已解析部分为左操作数,右侧递归解析为右操作数,组合成 AndExpression
  • 用栈或智能指针管理内存(推荐 std::unique_ptr)避免泄漏

注意事项与适用边界

解释器模式不是万能的,需注意:

  • 语法越复杂,类爆炸越严重 —— 超过 5–6 类非终结符就应考虑用 ANTLR 或手写 LL(1) 解析器替代
  • 性能敏感场景慎用 —— 每次执行都走虚函数调用+对象创建,不如编译为字节码或直接生成 C++ 代码
  • 调试困难 —— 表达式树结构隐含在对象关系中,建议添加 toString() 辅助调试
  • 上下文传递要精简 —— 避免把整个环境传入,只传必要数据(如变量表、作用域链)

它真正擅长的是“小而活”:规则常变、语法可控、团队需快速定制逻辑(如运营后台的用户筛选条件)。


# 字节  #   # c++  # 作用域  # 封装  # 标识符  # 字符串  # 递归  # 指针  # 继承  # 虚函数  # 接口  # 对象  # 布尔  # 适用于  # 的是  # 不高  # 可将  # 第三方  # 建立一个  # 都走  # 应考虑 


相关文章: 重庆市网站制作公司,重庆招聘网站哪个好?  建站主机选购指南:核心配置与性价比推荐解析  网站制作话术技巧,网站推广做的好怎么话术?  学校为何禁止电信移动建设网站?  建站主机如何选?高性价比方案全解析  如何在Golang中使用replace替换模块_指定本地或远程路径  C#怎么创建控制台应用 C# Console App项目创建方法  香港服务器网站推广:SEO优化与外贸独立站搭建策略  如何在建站宝盒中设置产品搜索功能?  如何确保西部建站助手FTP传输的安全性?  建站为何优先选择香港服务器?  如何通过IIS搭建网站并配置访问权限?  利用JavaScript实现拖拽改变元素大小  如何在服务器上三步完成建站并提升流量?  建站之星如何保障用户数据免受黑客入侵?  C++用Dijkstra(迪杰斯特拉)算法求最短路径  如何在万网主机上快速搭建网站?  如何通过虚拟主机快速完成网站搭建?  建站之星导航菜单设置与功能模块配置全攻略  小型网站建站如何选择虚拟主机?  ,怎么在广州志愿者网站注册?  logo在线制作免费网站在线制作好吗,DW网页制作时,如何在网页标题前加上logo?  实现虚拟支付需哪些建站技术支撑?  如何通过主机屋免费建站教程十分钟搭建网站?  如何使用Golang安装API文档生成工具_快速生成接口文档  如何快速搭建FTP站点实现文件共享?  建站之星如何修改网站生成路径?  制作农业网站的软件,比较好的农业网站推荐一下?  建站之星安装后如何自定义网站颜色与字体?  mc皮肤壁纸制作器,苹果平板怎么设置自己想要的壁纸我的世界?  宝塔建站助手安装配置与建站模板使用全流程解析  上海网站制作网页,上海本地的生活网站有哪些?最好包括生活的各个方面的?  建站ABC备案流程中有哪些关键注意事项?  简历在线制作网站免费版,如何创建个人简历?  百度网页制作网站有哪些,谁能告诉我百度网站是怎么联系?  建站之星安全性能如何?防护体系能否抵御黑客入侵?  如何快速搭建虚拟主机网站?新手必看指南  c# 在高并发场景下,委托和接口调用的性能对比  如何在阿里云虚拟服务器快速搭建网站?  如何在万网ECS上快速搭建专属网站?  PHP正则匹配日期和时间(时间戳转换)的实例代码  php条件判断怎么写_ifelse和switchcase的使用区别【对比】  在线制作视频网站免费,都有哪些好的动漫网站?  如何选择服务器才能高效搭建专属网站?  建站主机选哪家性价比最高?  TestNG的testng.xml配置文件怎么写  微网站制作教程,我微信里的网站怎么才能复制到浏览器里?  建站之星如何快速解决建站难题?  如何用免费手机建站系统零基础打造专业网站?  如何通过VPS建站无需域名直接访问? 

您的项目需求

*请认真填写需求信息,我们会在24小时内与您取得联系。