首页 > 其他分享 >设计模式(行为型设计模式——解释器模式)

设计模式(行为型设计模式——解释器模式)

时间:2024-03-20 13:31:39浏览次数:24  
标签:Node 解释器 right 模式 value interpreter 设计模式 public left

设计模式(行为型设计模式——解释器模式)

解释器模式

基本定义

给分析对象定义一个语言,并定义该语言的文法表示,设计一个解析器来解释语言中的句子。

模式结构

  • AbstractExpression:抽象表达式。声明一个抽象的解释操作,该接口为抽象语法树中所有的节点共享。

  • TerminalExpression:终结符表达式。现与文法中的终结符相关的解释操作。实现抽象表达式中所要求的方法。解释结果。

  • NonterminalExpression:非终结符表达式。为文法中的非终结符相关的解释操作。一般情况下以子类的形式出现。

  • Context:环境类。包含解释器之外的一些全局信息。

  • Client: 客户端类

代码实现

AbstractExpression:抽象解释器
public interface Node {
    int interpreter();
}
TerminalExpression:终结符表达式
/**
* 返回最终值
*/
public class ValueNode implements Node {


    private int value;


    public ValueNode(int value) {
        this.value = value;
    }


    @Override
    public int interpreter() {
        return this.value;
    }
}
NonterminalExpression:非终结符表达式
public class SymbolNode implements Node{

    Node left;
    Node right;

    public SymbolNode(Node left, Node right) {
        this.left = left;
        this.right = right;
    }

    @Override
    public int interpreter() {
        return 0;
    }
}
/**
* 加法
*/
public class AddNode extends SymbolNode {

    public AddNode(Node left, Node right) {
        super(left, right);
    }

    @Override
    public int interpreter() {
        return left.interpreter() + right.interpreter();
    }
}
/**
* 减法
*/
public class SubtractNode extends SymbolNode {

    public SubtractNode(Node left, Node right) {
        super(left, right);
    }

    @Override
    public int interpreter() {
        return left.interpreter() - right.interpreter();
    }
}
/**
* 乘法
*/
public class MulNode extends SymbolNode {

    public MulNode(Node left, Node right) {
        super(left, right);
    }

    @Override
    public int interpreter() {
        return left.interpreter() * right.interpreter();
    }
}
/**
* 除法
*/
public class DivisionNode extends SymbolNode {

    public DivisionNode(Node left, Node right) {
        super(left, right);
    }

    @Override
    public int interpreter() {
        return left.interpreter() / left.interpreter();
    }
}
Context:环境类
public class Calculator {


    private Node node;


    private String statement;


    public void build(String statement){
        Node left = null;
        Node right = null;
        //提供容器,存储关系
        Stack stack = new Stack();
        //重点,将node存入stack, 存储前需确认表达式的顺序和执行结果
        //以空格分隔
        String[] statementArr = statement.split(" ");
        int index = 0;
        int value;
        for (String state : statementArr){
            index++;
            switch (state){
                case "*" ://乘法
                    left = (Node)stack.pop();
                    value = Integer.parseInt(statementArr[index]);
                    right = new ValueNode(value);
                    stack.push(new MulNode(left, right));
                    break;


                case "/" : //除法
                    left = (Node)stack.pop();
                    value = Integer.parseInt(statementArr[index]);
                    right = new ValueNode(value);
                    stack.push(new DivisionNode(left, right));
                    break;
                    
                case "+" : //加法
                    left = (Node)stack.pop();
                    value = Integer.parseInt(statementArr[index]);
                    right = new ValueNode(value);
                    stack.push(new AddNode(left, right));
                    break;


                case "-" : //减法
                    left = (Node)stack.pop();
                    value = Integer.parseInt(statementArr[index]);
                    right = new ValueNode(value);
                    stack.push(new SubtractNode(left, right));
                    break;


                default:
                    stack.push(new ValueNode(Integer.parseInt(state)));
                    break;
            }


        }
        //stack pop显示栈顶元素并且移除栈顶元素
        this.node = (Node) stack.pop();
    }


    public int compute(){
        return node.interpreter();
    }


}
Client: 客户端类
@Slf4j
public class Test {
    public static void main(String[] args){
    //约定以空格分割
        String statement = "3 * 2 / 2 + 1 - 2";
        Calculator calculator = new Calculator();
        calculator.build(statement);
        int result = calculator.compute();
        log.info("statement : {} ------> value = {}", statement, result);
    }
}

优点

  • 扩展性好。由于在解释器模式中使用类来表示语言的文法规则,因此可以通过继承等机制来改变或扩展文法。

  • 容易实现。在语法树中的每个表达式节点类都是相似的,所以实现其文法较为容易

缺点

  • 执行效率比较低,可利用场景比较少。

  • 解释器模式中的每条规则至少需要定义一个类,当包含的文法规则很多时,类的个数将急剧增加,导致系统难以管理与维护。

  • 可应用的场景比较少。在软件开发中,需要定义语言文法的应用实例非常少,所以这种模式很少被使用到。

使用场景

  • 当一个语言需要解释执行,并且语言中的句子可以表示为一个抽象语法树的时候,如 XML 文档解释。

  • 当问题重复出现,且可以用一种简单的语言来进行表达时。

  • 当语言的文法较为简单,且执行效率不是关键问题时。

模式总结

  • 由于在解释器模式中使用类来表示语言的文法规则,因此可以通过继承等机制来改变或扩展文法。

  • 虽然解释器的可扩展性强,但是如果语法规则的数目太大的时候,该模式可能就会变得异常复杂。所以解释器模式适用于文法较为简单的。

标签:Node,解释器,right,模式,value,interpreter,设计模式,public,left
From: https://blog.csdn.net/weixin_44675056/article/details/136839463

相关文章

  • 设计模式(行为型设计模式——命令模式)
    设计模式(行为型设计模式——命令模式)命令模式基本定义将一个请求封装为一个对象,使发出请求的责任和执行请求的责任分割开。这样两者之间通过命令对象进行沟通,这样方便将命令对象进行储存、传递、调用、增加与管理。模式结构Command:抽象命令类ConcreteCommand:具体命......
  • shell编程有哪些常用解释器?
    在Shell编程中,常用的解释器(也称为Shell)包括以下几种:1.Bash(BourneAgainSHell):Bash是最常用的Shell解释器之一,也是许多Linux系统的默认Shell。它提供了许多强大的功能,如命令行编辑、命令历史、命令别名等。Bash的语法与BourneShell(sh)兼容,但增加了一些扩展和特性。2.sh(Bourne......
  • Vue的MVVM模式与双向绑定原理
    v-model 是Vue.js框架中用于实现双向数据绑定的指令。它充分体现了MVVM(Model-View-ViewModel)模式中的双向数据绑定特性。下面我们将详细解释 v-model 如何体现MVVM和双向绑定:1.MVVM模式MVVM模式是一种软件架构设计模式,它将应用程序分为三个部分:Model(模型):代表应用程......
  • Android ART编译模式解析
    前言ART实际就是Androidruntime的缩写,他是Android版本新的虚拟机诞生ART使用预先(AOT)编译,并且从Android7.0(代号Nougat,简称N)开始结合使用AOT、即时(JIT)编译和配置文件引导型编译。区别1.预先编译ART模式与Dalvik模式最大的不同在于,在启用ART模式后,系统在......
  • 单例模式
    TS实现单例模式classAxios{//由于instance是一个静态属性,它会在整个应用程序的生命周期内保持存在,除非显式地将其设置为null或通过其他方式清除它的引用。privatestaticinstance:Axios|null=nullurl:stringtimeout:number//通过把构造函数设为私......
  • 组合设计模式Java代码快速开始
    组合模式介绍使用组合模式可以让用户可以使用统一的方式处理整个树形结构的个别对象和组合对象,从而简化客户端的操作。并且扩展性好当需要处理的对象是树形结构时可以考虑使用组合模式。节点和叶子节点存在很大差异的情况下不建议使用组合模式。代码举例不使用组合模式举例......
  • 代理模式(Proxy Pattern)__浅谈与装饰器模式的区别
    代理模式(英语:ProxyPattern)为其他对象提供一种代理以控制对这个对象的访问。 代理模式和装饰器模式的共同点:  不改变原有组件的情况下增强其功能 代理模式和装饰器模式的差异:  代理模式可以实现延迟加载,即在需要时才真正创建原有组件,而不是在创建代理时就立即创建原有......
  • Java类和接口的工厂模式
    作者平时学习的小总结,希望可以帮到你更好了解。本章节介绍Java类和接口的工厂模式,帮助你对工厂模式有一个新的了解,这三种工厂模式的升级基于前一代的基础上进行升级的,所以一步一步看下去效果会更好。1.简单工厂模式简单工厂模式就是我们把一些类(比如说某一产品汽车,有很多品种看......
  • Selenium Headless模式:无头浏览器的使用与优势
    简介在现代Web开发和测试中,自动化工具的应用变得越来越重要。Selenium作为一种流行的自动化测试工具,为开发者提供了强大的功能来模拟用户行为和进行网页测试。其中,Selenium的Headless模式,即无头浏览器,为开发者提供了一种更高效、更隐秘的测试方式。本文将探讨SeleniumHeadless......
  • 探索发布-订阅模式的深度奥秘-实现高效、解耦的系统通信
    ​......