03-JAVA设计模式-解析器模式

解释器模式

什么是解析器模式

在Java中,解释器模式(Interpreter Pattern)是一种行为设计模式,它给定一个语言,定义它的文法的一种表示,并定义一个解释器,该解释器使用该表示来解释语言中的句子。解释器模式主要用于处理包含特定语法规则的简单语言,如数学表达式、SQL语句、配置文件等。

解释器模式主要由以下角色组成:

  • 抽象表达式(Abstract Expression): 声明一个解释操作的接口,这个接口叫做解释操作。
  • 终结符表达式(Terminal Expression): 实现抽象表达式接口,该接口表示处理解释器语法中出现的终结符,终结符是语法中不可再分的最小单位。
  • 非终结符表达式(Nonterminal Expression): 同样实现抽象表达式接口,该接口表示处理解释器语法中出现的非终结符,非终结符是语法中的短语或句子,一般由终结符构成。
  • 环境角色(Context): 包含解释器之外的全局信息,一般是用来存储有关被解释句子的信息,如符号表的值等。
  • 客户端(Client): 构建(或给定)表示该文法的抽象语法树,并调用解释器的解释方法来执行程序。

注意:

解析器模式是一种不常用的设计模式

用于描述如何构成已个简单的语言解释器,主要用于使用面相对象语言开发编译器和解释器设计

当我们需要开发一种新的语言是,可以考虑使用

实际开发中尽量不要使用解释器模式,后期维护忽悠很大的麻烦,在项目中,可以使用Jruby,Groovy,java的js引擎来替代解释器的作用,弥补java语言的不足

案例

实现通过输入字符串10+30-25计算得到最终的结果

UML

在这里插入图片描述

实现步骤:

  • 定义解释接口及返回值Expression
  • 创建具体的表达式-数字NumberExpression
  • 创建具体的表达式-符号OperatorExpression
  • 创建具体的表达式-加法AddExpression
  • 创建具体的表达式-减法SubtractionExpression
  • 创建解析类ExpressionParser,实现表达式解析,且用队列存储解析后的表达式,提供interpret()方法执行表达式

实现代码

Expression.java

// 抽象表达式接口
public interface Expression {// 定义解释接口及返回值int interpret();
}

NumberExpression.java

// 具体的表达式-数字
public class NumberExpression implements Expression{// 定义接收值private Integer value;public NumberExpression(Integer value) {this.value = value;}@Overridepublic int interpret() {return value;}
}

OperatorExpression.java

// 具体的表达式-符号
public class OperatorExpression implements Expression{// 定义接收值private char value;public OperatorExpression(char value) {this.value = value;}@Overridepublic int interpret() {return value;}
}

AddExpression.java

// 具体的表达式-加法
public class AddExpression implements Expression{// 解析器左侧表达式private Expression left;// 解析器右侧表达式private Expression right;public AddExpression(Expression left, Expression right) {this.left = left;this.right = right;}@Overridepublic int interpret() {return left.interpret() + right.interpret();}
}

Expression.java

// 具体的表达式-减法
public class SubtractionExpression implements Expression{// 解析器左侧表达式private Expression left;// 解析器右侧表达式private Expression right;public SubtractionExpression(Expression left, Expression right) {this.left = left;this.right = right;}@Overridepublic int interpret() {return left.interpret() - right.interpret();}
}

ExpressionParser.java

import java.util.ArrayDeque;
import java.util.Deque;// 解析器类
public class ExpressionParser{private String expression;private int index = 0;private Deque<Expression> queue = new ArrayDeque<Expression>();public ExpressionParser(String expression) {this.expression = expression;parse();}// 解析表达式public void parse(){// 判断是否连续数字boolean markDigit = false;// 判断是否连续运算符boolean markOperation = false;while (index < expression.length()){char c = expression.charAt(index);// 字符判断if(Character.isDigit(c) && !markDigit){markDigit = true;markOperation = false;parseNumber();continue;}else if((c == '+' || c == '-') && !markOperation){markDigit = false;markOperation = true;parseOperation();} else {throw new RuntimeException("输入字符错误: " + c);}index++;}}// 解析数字private void parseNumber() {StringBuilder sb = new StringBuilder();while (index < expression.length() && Character.isDigit(expression.charAt(index))) {sb.append(expression.charAt(index));index++;}int number = Integer.parseInt(sb.toString());queue.offer(new NumberExpression(number));}// 解析符号private void parseOperation() {queue.offer(new OperatorExpression(expression.charAt(index)));}// 计算public int interpret() {Expression result = null;int len = queue.size();for (int i = 0; i < len; i++) {Expression pop = queue.poll();if(result == null){result = pop;}else if (pop instanceof OperatorExpression &&  pop.interpret() == '+') {result = new AddExpression(result, queue.poll());i++;}else if (pop instanceof OperatorExpression &&  pop.interpret() == '-') {result = new SubtractionExpression(result, queue.poll());i++;}else {throw new RuntimeException("表达式异常");}}return result.interpret();}
}

TestClient.java

// 测试
public class TestClient {public static void main(String[] args) {String expression = "10+30-25";ExpressionParser parser = new ExpressionParser(expression);System.out.println("Result: " + parser.interpret());}
}

执行结果:

在这里插入图片描述

gitee源码

git clone https://gitee.com/dchh/JavaStudyWorkSpaces.git

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

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

相关文章

Linux提权--SUID提权内核漏洞本地用户提权

免责声明:本文仅做技术交流与学习,请不要乱搞破坏... 目录 SUID提权 漏洞成因 提权过程: 手工命令探针: 参考利用&#xff1a; 脚本探针: LinEnum.sh traitor linuxprivchecker等等... Linux命令的利用: find命令 利用nc反弹 利用python反弹--棱角 内核漏洞本地用…

电磁兼容(EMC):生产的ESD防护要点及措施

目录 1. 接地和连接系统 2. 操作员及工作区域 3. 地板 4. 座椅 5. 防静电车间 早期电子产品出现质量问题有80%的问题都是生产过程静电所引起的。随着ESD的管理程序系统的普及&#xff0c;ESD问题相当减小很多。例如当今的S20.20静电控制程序产生于IBM内部审核和控制系统。…

打不完!真的打不完!海量用户线索车企该怎么办?用AI!

当车企面临海量用户线索&#xff0c;怎么找到精准用户&#xff1f;大量的电话根本打不完&#xff0c;这种情况怎么办&#xff1f;建议借助AI&#xff0c;降本增效。下面以某车企为例来帮助大家解决这个难题&#xff01; 某车企面临的问题主要有三点 第一&#xff1a;车企有来自…

罗德与施瓦茨矢量网络分析仪ZNB20相位一致性

矢量网络分析仪(VNA)是电子测量领域中非常重要的一类仪器,广泛应用于射频和微波电路的测试与分析。其中,德国罗德与施瓦茨公司生产的ZNB20型号是一款性能出色的矢量网络分析仪,深受业内人士的青睐。本文将重点介绍ZNB20在相位测量方面的特点和优势,为用户提供全面的使用参考。 …

微信红包架构

文章目录 包发抢拆抢红包——拆包算法——要解决并发问题那怎么做呢&#xff1f;——预分配红包预分配怎么实现呢&#xff1f;redis集群解决日均百亿级但微信没有用预分配方案哦——内存消耗过大——cas查询红包领取记录会很频繁的查询&#xff0c;从redis中查询hash&#xff0…

Blob对象实现文件下载

首先&#xff0c;要有下载按钮 <a download onClick{()> downloadAttentment(data)}>下载</a>其次&#xff0c;定义下载function // 此处去处理 blob 对象 const downloadStreamByATag (data, suffix, name) > {if (suffix void 0) { suffix xls; }if (…

Python数据结构与算法(1):将序列分解为单独的变量

问题 现在有一个包含 N 个元素的元组或者是序列&#xff0c;怎样将它里面的值解压后同时赋值给 N 个变量&#xff1f; 解决方案 任何的序列&#xff08;或者是可迭代对象&#xff09;可以通过一个简单的赋值操作来分解为单独的变量。 唯一的要求就是变量的总数和结构必须与序…

一个人可能代表一群人

最近有个小伙伴问他是做货代的&#xff0c;怎么能找到自己的客户&#xff1f;刚看到这个问题的时候&#xff0c;觉得这个不属于自己的范围&#xff0c;因为自己不是做货代的&#xff0c;怎么知道他们怎么工作&#xff0c;怎么去搜索客户呢&#xff1f; 但是仔细想来&#xff0…

【Java EE】日志框架(SLF4J)与门面模式

文章目录 &#x1f340;SLF4j&#x1f333;门面模式(外观模式)&#x1f338;门面模式的定义&#x1f338;门面模式的模拟实现&#x1f338;门面模式的优点 &#x1f332;关于SLF4J框架&#x1f338;引入日志门面 ⭕总结 &#x1f340;SLF4j SLF4J不同于其他⽇志框架,它不是⼀个…

跟TED演讲学英文:AI isn‘t as smart as you think -- but it could be by Jeff Dean

AI isn’t as smart as you think – but it could be Link: https://www.ted.com/talks/jeff_dean_ai_isn_t_as_smart_as_you_think_but_it_could_be Speaker: Jeff Dean Jeffrey Adgate “Jeff” Dean (born July 23, 1968) is an American computer scientist and software…

【服务器部署篇】Linux下快速安装Jenkins

作者介绍&#xff1a;本人笔名姑苏老陈&#xff0c;从事JAVA开发工作十多年了&#xff0c;带过刚毕业的实习生&#xff0c;也带过技术团队。最近有个朋友的表弟&#xff0c;马上要大学毕业了&#xff0c;想从事JAVA开发工作&#xff0c;但不知道从何处入手。于是&#xff0c;产…

网站建设企业网站优化

近年来&#xff0c;随着互联网的迅速发展&#xff0c;企业网站已经成为了企业展示自我形象与实力的重要载体之一。然而&#xff0c;单单拥有一个美观、简洁的企业网站并不能让企业在竞争激烈的市场中脱颖而出。因此&#xff0c;在建设企业网站的过程中&#xff0c;我们需要将企…

C++常用的输入输出方法(ACM模式)

文章目录 前言一、输入输出方法1、cin2、getline()3、getchar() 二、算法案例1、一维数组1.1 输入固定长度1.2长度不固定 2、固定二维数组3、以非空格隔开的元素输入3、常见数据结构定义以及输入3.1 链表 前言 C中的输入输出函数有很多&#xff0c;我们本章只针对大部分算法题…

上位机图像处理和嵌入式模块部署(树莓派4b开机界面程序自启动)

【 声明&#xff1a;版权所有&#xff0c;欢迎转载&#xff0c;请勿用于商业用途。 联系信箱&#xff1a;feixiaoxing 163.com】 前面我们学习了如何在树莓派4b上面开发qt&#xff0c;也学习了如何用/etc/rc.local启动控制台程序&#xff0c;那今天我们继续学习一下如何利用树莓…

docker compose安装redis

一、安装准备 在docker hub查看redis镜像版本。查看地址如下&#xff1a; Dockerhttps://hub-stage.docker.com/_/redis/tags 二、拉取docker镜像 我这里用redis:6.2.14版本&#xff0c;先拉取镜像。命令如下&#xff1a; docker pull redis:6.2.14 查看刚刚下载的镜像&am…

[python数据处理系列] 深入理解与实践基于聚类的过采样与欠采样技术:以K-Means为例

目录 一、过采样介绍 (一)什么是过采样 (二)过采样的优点 (三)过采样的缺点 二、欠采样介绍 (一)什么是欠采样 (二)欠采样的优点 (三)欠采样的缺点 三、基于聚类的欠抽样方法(K-Means欠采样/KMeans-Undersampling) (一)KMeans欠采样原理及其步骤介绍 (二)为什么不采…

clickhouse学习笔记05

ClickHouseSpringBoot2.XMybatisPlus整合搭建 添加需要的依赖&#xff1a; 添加clickhouse依赖&#xff1a; 配置数据库配置&#xff1a; 我们框架就搭建完了。 ClickHouse的项目案例统计需求讲解 ClickHouse的项目案例统计库表和数据准备 添加数据&#xff1a; 数据都插入进来…

js逆向进阶篇-某团酒店

提示!本文章仅供学习交流,严禁用于任何商业和非法用途,未经许可禁止转载,禁止任何修改后二次传播,擅自使用本文讲解的技术而导致的任何意外,作者均不负责,如有侵权,可联系本文作者删除! 案例分析: 先来看看请求中有哪些参数是需要我们逆向,如下: mtgsig、fp、roh…

ERROR: [7df2405] missing Change-Id in commit message footer

git push origin HEAD:refs/for/[分支名] 使用“git push origin HEAD:refs/for/[分支名]”,报错信息见下图 报错信息:ERROR: [7df2405] missing Change-Id in commit message footer 解决办法 根据git的提示依次执行下面的语句即可 第一步:输入下面的语句,下载一个 g…

Java---数据类型与变量

1.字面常量 字面常量就是我们经常所说的常量&#xff0c;常量即在程序运行期间&#xff0c;固定不变的量。且常量是无法改变的&#xff0c;如果我们的代码有改变常量的操作&#xff0c;程序就会报错。 1.1字面常量的分类 字符串常量&#xff0c;整型常量&#xff0c;浮点数常…