对解释器模式的理解

news2025/4/11 13:37:09

对解释器模式的理解

    • 一、场景
      • 1、题目【[来源](https://kamacoder.com/problempage.php?pid=1096)】
        • 1.1 题目描述
        • 1.2 输入描述
        • 1.3 输出描述
        • 1.4 输入示例
        • 1.5 输出示例
    • 二、不采用解释器模式
      • 1、代码
      • 2、“缺点”
    • 三、采用解释器模式
      • 1、代码
      • 2、“优点”
    • 四、思考
      • 1、解释器模式的意义

一、场景

1、题目【来源】

1.1 题目描述

小明正在设计一个计算器,用于解释用户输入的简单数学表达式,每个表达式都是由整数、加法操作符+、乘法操作符组成的,表达式中的元素之间用空格分隔,请你使用解释器模式帮他实现这个系统。

1.2 输入描述

每行包含一个数学表达式,表达式中包含整数、加法操作符(+)和乘法操作符(*)。 表达式中的元素之间用空格分隔。

1.3 输出描述

对于每个输入的数学表达式,每行输出一个整数,表示对应表达式的计算结果。

1.4 输入示例
2 + 3
5 * 2
3 + 4 * 2
1.5 输出示例
5
10
11

二、不采用解释器模式

1、代码

  • 计算器
public class Calculator {
    private final Deque<Integer> numberStack;
    private final Deque<String> operatorStack;

    public Calculator() {
        numberStack = new ArrayDeque<>();
        operatorStack = new ArrayDeque<>();
    }

    public Integer calculate(String expression) {
        if (expression == null || expression.isEmpty()) {
            return null;
        }


        String[] tokens = expression.trim().split("\\s+");
        for (String token : tokens) {
            if (isOperator(token)) {
                while (!operatorStack.isEmpty() && ("*".equals(operatorStack.peek()) && "+".equals(token))) {
                    numberStack.push(numberStack.pop() * numberStack.pop());
                }

                operatorStack.push(token);

            } else {
                numberStack.push(Integer.parseInt(token));
            }
        }

        while (!operatorStack.isEmpty()) {
            Integer num2 = numberStack.pop();
            Integer num1 = numberStack.pop();
            String operator = operatorStack.pop();

            if ("+".equals(operator)) {
                numberStack.push(num1 + num2);
            } else if ("*".equals(operator)) {
                numberStack.push(num1 * num2);
            }
        }


        return numberStack.pop();
    }

    private static boolean isOperator(String s) {
        return "+".equals(s) || "*".equals(s);
    }
}
  • 客户端
public class Main {
    public static void main(String[] args) {
        Scanner scanner = new Scanner(System.in);
        Calculator calculator = new Calculator();

        while (scanner.hasNextLine()) {
            String expression = scanner.nextLine();
            System.out.println(calculator.calculate(expression));
        }
    }
}

2、“缺点”

  • 倒反天罡了,我觉得不采用解释器模式反而更好。😃

三、采用解释器模式

1、代码

  • 定义表达式
public interface Expression {
    int interpret();
}

public class NumberExpression implements Expression {
    private final int number;

    public NumberExpression(int number) {
        this.number = number;
    }

    @Override
    public int interpret() {
        return number;
    }
}

public class AddExpression implements Expression {
    private final Expression left;
    private final Expression right;

    public AddExpression(Expression left, Expression right) {
        this.left = left;
        this.right = right;
    }

    @Override
    public int interpret() {
        return left.interpret() + right.interpret();
    }
}

public class MultiplyExpression implements Expression {
    private final Expression left;
    private final Expression right;

    public MultiplyExpression(Expression left, Expression right) {
        this.left = left;
        this.right = right;
    }

    @Override
    public int interpret() {
        return left.interpret() * right.interpret();
    }
}
  • 计算器
public class Calculator {
    private final Deque<Expression> expressionStack = new ArrayDeque<>();
    private final Deque<String> operatorStack = new ArrayDeque<>();

    public Integer calculate(String expression) {
        if (expression == null || expression.isEmpty()) {
            return null;
        }

        String[] tokens = expression.trim().split("\\s+");
        for (String token : tokens) {
            if (isOperator(token)) {
                // 处理操作符优先级
                while (!operatorStack.isEmpty() && ("*".equals(operatorStack.peek()) && "+".equals(token))) {
                    processOperator();
                }
                operatorStack.push(token);
            } else {
                expressionStack.push(new NumberExpression(Integer.parseInt(token)));
            }
        }

        // 处理剩余的操作符
        while (!operatorStack.isEmpty()) {
            processOperator();
        }

        // 最终栈中只剩下一个表达式对象
        return expressionStack.pop().interpret();
    }

    private void processOperator() {
        Expression right = expressionStack.pop();
        Expression left = expressionStack.pop();
        String operator = operatorStack.pop();

        Expression operation;
        if ("+".equals(operator)) {
            operation = new AddExpression(left, right);
        } else if ("*".equals(operator)) {
            operation = new MultiplyExpression(left, right);
        } else {
            throw new IllegalArgumentException("Unknown operator: " + operator);
        }

        expressionStack.push(operation);
    }

    private static boolean isOperator(String s) {
        return "+".equals(s) || "*".equals(s);
    }
}
  • 客户端代码和之前一样

2、“优点”

  • 有种画蛇添足的感觉,貌似没有体现解释器模式的优势。

四、思考

1、解释器模式的意义

  • 抽丝剥茧,从一段简单的代码说起:

public class InterpretTest {
    public static void main(String[] args) {
        // 3 + 4 * 2
        Expression expression = new AddExpression(new NumberExpression(3),
                new MultiplyExpression(new NumberExpression(4), new NumberExpression(2)));

        System.out.println(expression.interpret());
    }
}
  • 当用户输入字符串“3 + 4 * 2”的时候,转换成Expression本身就比较复杂,但一旦转换好了,如上所示,Expression就发挥优势了。

  • 因此,当我们需要解释语法规则时,应该是分为2步的:

    • (1)将字符串类型的表达式,转成Expression对象。
    • (2)Expression对象解释执行。【这一步才体现了解释模式的价值】

    如果不采用解释模式,实际上如同“二、不采用解释器模式”的代码所示,是一边解析表达式字符串,一边执行的。

  • 如果解释执行本身就比较简单(计算器的例子便是如此),先转成Expression对象再解释执行的意义确实不大。

  • 但如果解释执行本身就比较复杂,而且如何解释执行可能变化,这时候解释器模式的价值就体现出来了。

    目前没在实战中遇到过,等遇到了感受就深了。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2329970.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

解决Oracle PL/SQL中“表或视图不存在“错误的完整指南

解决Oracle PL/SQL中"表或视图不存在"错误的完整指南 前言问题概述根本原因分析一、 编译时与运行时验证差异二、权限问题三、 Schema命名问题 实际案例演示案例1&#xff1a;动态分表查询案例2&#xff1a;权限不足的场景 实用排查步骤排查流程图最佳实践建议解决方…

SSH远程连接服务器(cursor)

安装Remote-SSH插件 Cursor是基于VSCode的&#xff0c;因此支持VSCode的Remote-SSH功能。打开Cursor&#xff0c;进入扩展市场&#xff08;左侧活动栏的“Extensions”图标&#xff09;。搜索“Remote - SSH”插件并安装&#xff08;由Microsoft提供&#xff09;。 配置SSH 在…

idea gitlab 操作

1.拉取脚本 账号登录 就可以获取git代码 2. 版本回退 hard暴力回退到暂存区 缓存区消失 3.版本合并 切换到目标分区 选择点击开发分区 进行合并

【MATLAB第113期】基于MATLAB的EFAST扩展傅里叶幅度敏感性分析方法(有目标函数)

【MATLAB第113期】基于MATLAB的EFAST扩展傅里叶幅度敏感性分析方法&#xff08;有目标函数&#xff09; 一、方法概述 扩展傅里叶幅度敏感性检验&#xff08;EFAST&#xff09;是一种基于频域分析的全局敏感性分析方法&#xff0c;能够同时评估模型参数的一阶敏感性&#xff…

Unity3D开发AI桌面精灵/宠物系列 【三】 语音识别 ASR 技术、语音转文本多平台 - 支持科大讯飞、百度等 C# 开发

Unity3D 交互式AI桌面宠物开发系列【三】ASR 语音识别 该系列主要介绍怎么制作AI桌面宠物的流程&#xff0c;我会从项目开始创建初期到最终可以和AI宠物进行交互为止&#xff0c;项目已经开发完成&#xff0c;我会仔细梳理一下流程&#xff0c;分步讲解。 这篇文章主要讲有关于…

Qt -信号与槽

博客主页&#xff1a;【夜泉_ly】 本文专栏&#xff1a;【暂无】 欢迎点赞&#x1f44d;收藏⭐关注❤️ 目录 前言引入connect调用链模板类型的connectQObject::connectImplQObjectPrivate::connectImpl qobject_p_p.hconnect作用总结ai对信号与槽的模拟实现 前言 面向对象&am…

Django中使用不同种类缓存的完整案例

Django中使用不同种类缓存的完整案例 推荐超级课程: 本地离线DeepSeek AI方案部署实战教程【完全版】Docker快速入门到精通Kubernetes入门到大师通关课AWS云服务快速入门实战目录 Django中使用不同种类缓存的完整案例步骤1:设置Django项目步骤2:设置URL路由步骤3:视图级别…

解锁健康密码,拥抱品质生活

在生活节奏不断加快的今天&#xff0c;健康养生已成为人们关注的焦点。它不仅关乎当下生活质量&#xff0c;更是对未来幸福的投资。从日常生活的点滴出发&#xff0c;掌握正确养生方法&#xff0c;我们就能轻松收获健康。​ 饮食是健康的基石。我们应当遵循 “食物多样&#x…

ABAP 新语法 - corresponding

在 ABAP 中&#xff0c;CORRESPONDING 操作符用于根据字段名称自动映射结构体&#xff08;Structure&#xff09;或内表&#xff08;Internal Table&#xff09;的字段值。它比传统的 MOVE-CORRESPONDING 语句更灵活&#xff0c;支持更多控制选项。 基础用法 data: begin of …

HTML零基础入门笔记:狂神版

前言 本笔记是学习狂神的java教程&#xff0c;建议配合视频&#xff0c;学习体验更佳。 【狂神说Java】HTML5完整教学通俗易懂_哔哩哔哩_bilibili 第1-2章&#xff1a;Java零基础入门笔记&#xff1a;(1-2)入门&#xff08;简介、基础知识&#xff09;-CSDN博客 第3章&…

FreeRTOS移植笔记:让操作系统在你的硬件上跑起来

一、为什么需要移植&#xff1f; FreeRTOS就像一套"操作系统积木"&#xff0c;但不同硬件平台&#xff08;如STM32、ESP32、AVR等&#xff09;的CPU架构和外设差异大&#xff0c;需要针对目标硬件做适配配置。移植工作就是让FreeRTOS能正确管理你的硬件资源。 二、…

c语言修炼秘籍 - - 禁(进)忌(阶)秘(技)术(巧)【第五式】动态内存管理

c语言修炼秘籍 - - 禁(进)忌(阶)秘(技)术(巧)【第五式】动态内存管理 【心法】 【第零章】c语言概述 【第一章】分支与循环语句 【第二章】函数 【第三章】数组 【第四章】操作符 【第五章】指针 【第六章】结构体 【第七章】const与c语言中一些错误代码 【禁忌秘术】 【第一式…

MySQL表的增删改查基础版

这一部分内容比较多&#xff0c;请大家结合目录查看&#x1f440; 增删改查 这一部分内容比较多&#xff0c;请大家结合目录查看&#x1f440; 一、新增1.插入2.指定列插入3.一次插入多行记录 二、查询1.全列查询2.指定列查询3.查询字段为表达式4.别名5.去重6.多列去重7.排序8.…

【备赛】蓝桥杯嵌入式实现led闪烁

原理 由于蓝桥杯的板子带有锁存器&#xff0c;并且与lcd屏幕有冲突&#xff0c;所以这个就成了考点。 主要就是用定时器来实现&#xff0c;同时也要兼顾lcd的冲突。 一、处理LCD函数 首先来解决与lcd屏幕冲突的问题&#xff0c;把我们所有用到的lcd函数改装一下。 以下是基…

【Python】贝叶斯,条件概率是怎么回事儿

【Python】贝叶斯&#xff0c;条件概率是怎么回事儿 一、原理简介1.1 贝叶斯定理1.2 朴素贝叶斯假设 二、算法实现过程2.1 数据准备与预处理2.2 模型训练与预测2.2.1 高斯朴素贝叶斯 - 对应连续型数据2.2.2 多项式朴素贝叶斯 - 离散型数据 2.3 模型评估 三、算法优缺点分析3.1 …

Flink介绍——实时计算核心论文之Storm论文详解

引入 我们通过以下两篇文章&#xff0c;深入探索了S4是如何抽象流式计算模型&#xff0c;如何设计架构和系统&#xff0c;存在那些局限&#xff1a; 论文详解论文总结 Yahoo推出的S4 并没有在历史舞台上站稳脚跟&#xff0c;在S4的论文发表的同一年&#xff0c;我们今天的主…

001 使用单片机实现的逻辑分析仪——吸收篇

本内容记录于韦东山老师的毕设级开源学习项目&#xff0c;含个人观点&#xff0c;请理性阅读。 个人笔记&#xff0c;没有套路&#xff0c;一步到位&#xff0c;欢迎交流&#xff01; 00单片机的逻辑分析仪与商业版FPGA的逻辑分析仪异同 对比维度自制STM32逻辑分析仪商业版逻…

11-产品经理-创建产品

在“产品”-“仪表盘”内&#xff0c;可以查看系统中关于产品及相关需求的统计。 在“产品”-“产品列表”页面&#xff0c;可以按项目集、项目查看其关联产品。还可以添加产品、编辑产品线、或者导出产品列表。 产品看板&#xff0c;通过看板方式查看产品、产品计划和产品下的…

低代码开发平台:飞帆制作网页并集成到自己的网页中

应用场景&#xff1a; 有时&#xff0c;我们的网页使用了某个模版&#xff0c;或者自己写的 html、css、javascript 代码。只是网页中的一部分使用飞帆来制作。这样的混合网页如何实现呢&#xff1f; 其实很容易&#xff0c;来体验一下飞帆提供的功能&#xff01; 还记得这个…

语法: result=log (x);

LOG( ) 语法: resultlog (x); 参数: x是一个浮点数; 返回值: result等于返回值,是一个浮点数; 功能: 该函数是用来计算浮点数x的自然对数(即ln x);如果x小于或等于0,或x太大,则行为没有定义; 注意:存在error挂起; 如果在编写程序里包含了errno.h头文件,则范围和等级…