面向对象的设计模式(十三),解释器模式

解释器模式,从字面上解释来说就是为一个文法(具有特定语法的形式的语句或表达式)构造解释器,这个解释器用来解释这个文法,使得这样的具有某种书写规则的文法能够表示特定的功能,这样的特定书写规则也就是通常所说的语法,如C/C++,Java,Python等计算机语言有自己的语法。还有,一些解释型语言如Python。它在执行的时候须要Python解释器,这也就是一种解释器。

定义:解释器模式为一些具有特定书写规则的文法编写解释器,从而使得它具有某种意义。

使用场景:

  1. 作为解释器

    解释具有特定语法的语句的功能或者意义。如解释具有特定书写规则的语句,在以下的这个样例中就是作为解释器来解释运算表达式。

  2. 作为翻译机或者译码器。如高级语言中的常量,编译的时候编译器系统会将常量替换成它代表的数一样,相似的我们也能够定义一些特定的具有某种意义的语句。然后用对应的解释器来解释器它。

假设又这么个表达式或语句a_+_b_-_c,我们想要知道它的意思,假如我们规定_符号用来分隔数字和运算符,那么上面的这个语句表达的意思就是计算a+b-c。这和学生考试的时候。它们规定1表示A,2表示B,3表示C,4表示D来传选择题答案一样的,相似于编码译码过程。事实上译码器也就是解释器,那么我们通过代码来解释上面表达式的意思。

代码实现:

抽象解释器(基类)

/**
 * 主要的解释器。是全部解释器的基类
 * @author lt
 *
 */
public abstract class BaseInterpreter<T> {

    /**
     * 抽象解释方法
     * @return
     */
    public abstract T interpret();
}

?抽取了解释器的共性

整数解释器(解释整数)

/**
 * 整数解释器
 * @author lt
 *
 */
public class NumberInterpreter extends BaseInterpreter<Integer>{

    private int num;

    public NumberInterpreter(int num){
        this.num = num;
    }

    @Override
    public Integer interpret() {
        return this.num; // 自己主动装箱拆箱
    }
}

?解释整数,直接返回这个整数。这里的整数也就是终结符

运算符解释器(基类

/**
 * 二目运算符操作解释器,也是一个基类,由于有好多的二目运算符
 * @author lt
 *
 */
public abstract class OperatorInterpreter extends BaseInterpreter<Integer>{

    protected BaseInterpreter<Integer> exp1;
    protected BaseInterpreter<Integer> exp2;

    public OperatorInterpreter(BaseInterpreter<Integer> exp1,BaseInterpreter<Integer> exp2){
        this.exp1 = exp1;
        this.exp2 = exp2;
    }

}

?解释二目运算符。须要两个整数,为非终结符解释器。

加法解释器(计算加法)

/**
 * 加法解释器。计算加法
 * @author lt
 *
 */
public class AdditionInterpreter extends OperatorInterpreter{

    public AdditionInterpreter(BaseInterpreter<Integer> exp1,
            BaseInterpreter<Integer> exp2) {
        super(exp1, exp2);
    }

    /**
     * 用来计算加法
     */
    @Override
    public Integer interpret() {
        return exp1.interpret() + exp2.interpret();
    }
}

减法解释器(计算减法)

/**
 * 减法计算器
 * @author lt
 *
 */
public class SubtractionInterpreter extends OperatorInterpreter{

    public SubtractionInterpreter(BaseInterpreter<Integer> exp1,
            BaseInterpreter<Integer> exp2) {
        super(exp1, exp2);
    }

    @Override
    public Integer interpret() {
        return exp1.interpret() - exp2.interpret();
    }
}

计算器,翻译a_+_b_-_c(计算表达式)

import java.util.Stack;

/**
 * 计算器
 * @author lt
 *
 */
public class Calculator {

    private Stack<BaseInterpreter<Integer>> mExpStack = new Stack<BaseInterpreter<Integer>>();

    public Calculator(String expression){
        // 声明两个BaseInterpreter<Integer>的暂时变量,由于计算必须要记录两个数
        BaseInterpreter<Integer> exp1,exp2;

        //  以符号_分隔,这是我们自己规定的
        String[] exps = expression.split("_");

        for(int i=0;i<exps.length;i++){
            switch (exps[i].charAt(0)) {
            case ‘+‘: // 加法
                exp1 = mExpStack.pop();
                exp2 = new NumberInterpreter(Integer.valueOf(exps[++i]));
                mExpStack.push(new AdditionInterpreter(exp1, exp2));
                break;
            case ‘-‘:
                exp1 = mExpStack.pop();
                exp2 = new NumberInterpreter(Integer.valueOf(exps[++i]));
                mExpStack.push(new SubtractionInterpreter(exp1, exp2));
                break;
            default: // 数字
                mExpStack.push(new NumberInterpreter(Integer.valueOf(exps[i])));
                break;
            }
        }
    }

    /**
     * 计算
     * @return
     */
    public int calculate(){
        return mExpStack.pop().interpret();
    }
}

?这个类用来翻译a_+_b_-_c等形式结构的语句的意思。这里的符号_是我规定用来分隔数字的,当然你也能够规定其它符号作为分隔符。

这个类的方法也非常easy,构造方法中先是将表达式按符号_分隔。得到一些运算符和数字,然后在依据分隔出来的字符串的第一个字符的类型推断是+运算符还是-运算符还是数字,假设是+运算符。那么就将存储的上一个数字弹出栈并记录到exp1,然后得到运算符后面的那个字符串(肯定是数字)并记录到变量exp2中。最后用加法解释器解释这两个变量记录的数字并压入栈,等下一次循环的时候弹出和下一个数字进行计算;假设是-运算符,那么和+运算符是一样的,仅仅只是用的是减法解释器;假设是数字,直接压入栈,整个过程是逐步计算的过程。calculate方法用来输出计算结果。

測试

public class Test {

    public static void main(String[] args) {
        String testStr = "1_+_34_-_10_+_50";
        Calculator calculator = new Calculator(testStr);
        System.out.println("result="+calculator.calculate());
    }
}

结果:

?能够看到我们定义的解释器成功解释了a_+_b_-_c这样的形式的语句,事实上这个解释器实现的功能相似译码的功能或者翻译的功能。只是话说回来,尽管是成功解释了那种形式的语句,可是也仅仅能计算整数的加减法。假设想要做其它运算或者其它类型的数字,如乘除法和浮点型,那么须要加入对应的解释器,但假设涉及到混合运算的时候,那复杂多了,还得考虑优先级。这个时候这样的模式可能就不适合了,也就是说解释器模式适合于简单的语句。

总结

长处:

  • 灵活的扩展性

    当我们对语法规则扩展延伸适合,仅仅须要加入对应的非终结符解释器(如上面的加减法解释器),并在构建抽象语法树时,使用到新添加的解释器对象(如加入减法解释器)进行详细的解释(计算减法)就可以,非常方便。非终结符也就是还没有结束的符号,如以下样例中的加法和减法解释器分别解释的加好和减号一样,它们是二目运算符。其两边肯定要两个数。

缺点:

  • 类数量膨胀,后期维护困难。由于对于每一条文法都对应至少一个解释器类,会产生大量的类,导致后期维护困难。同一时候,对于复杂的文法,构建其抽象的语法树会显得比較繁琐,甚至须要构建多颗语法树,因此。对于复杂的文法并不推荐使用解释器模式。

原文地址:https://www.cnblogs.com/zhchoutai/p/8905653.html

时间: 2024-11-18 06:52:27

面向对象的设计模式(十三),解释器模式的相关文章

设计模式之解释器模式(Interpreter)摘录

23种GOF设计模式一般分为三大类:创建型模式.结构型模式.行为模式. 创建型模式抽象了实例化过程,它们帮助一个系统独立于如何创建.组合和表示它的那些对象.一个类创建型模式使用继承改变被实例化的类,而一个对象创建型模式将实例化委托给另一个对象.创建型模式有两个不断出现的主旋律.第一,它们都将关于该系统使用哪些具体的类的信息封装起来.第二,它们隐藏了这些类的实例是如何被创建和放在一起的.整个系统关于这些对象所知道的是由抽象类所定义的接口.因此,创建型模式在什么被创建,谁创建它,它是怎样被创建的,以

大话设计模式_解释器模式(Java代码)

解释器模式:给定一个语言,定义它的文法的一种表示,并定义一个解释器,这个解释器使用该表示来解释语言中的句子. 简单描述:一个AbstractExpression类,多个子类,存在一个Interpret方法,转义Context对象的信息.客户端根据信息实例化不同的Expression类,并调用其转义方法(这个过程可以使用简单工厂+反射进行) 大话设计模式中的截图: 代码例子: 假设HTML代码解释器: (1)第一类标签<HTML>(开始)/<HEAD>(头信息)/<BODY&g

GOF23设计模式之解释器模式和访问器模式的理解

设计模式之解释器模式Interpreter      是一种不常用的设计模式      用于描述如何构成一个简单的语言解释器,主要用于使用面向对象语言开发的编译器和解释器设计.      当我们需要开发一种新的语言时,可以考虑使用解释器模式.      尽量不要使用解释器模式,后期维护会有很大麻烦.在项目中可以使用jruby,Groovy,java的js引擎来代替解释器的作用,弥补java语言的不足. 开发中常见的场景:      EL表达式的处理      正则表达式解释器      SQL语

设计模式 ( 十三 ) 命令模式Command(对象行为型)

设计模式 ( 十三 ) 命令模式Command(对象行为型) 1.概述 在软件设计中,我们经常需要向某些对象发送请求,但是并不知道请求的接收者是谁,也不知道被请求的操作是哪个,我们只需在程序运行时指定具体的请求接收者即可,此时,可以使用命令模式来进行设计,使得请求发送者与请求接收者消除彼此之间的耦合,让对象之间的调用关系更加灵活. 例子1:电视机遥控器 : 遥控器是请求的发送者,电视机是请求的接收者,遥控器上有一些按钮如开,关,换频道等按钮就是具体命令,不同的按钮对应电视机的不同操作. 2.问题

C#设计模式:解释器模式(Interpreter Pattern)

一,C#设计模式:解释器模式(Interpreter Pattern) 1,解释器模式的应用场合是Interpreter模式应用中的难点,只有满足“业务规则频繁变化,且类似的模式不断重复出现,并且容易抽象为语法规则的问题”才适合使用解释器模式2,解释器设计模式每个解释的类有自己的规则,并且与其他业务规则不冲突 二,如下代码 using System; using System.Collections.Generic; using System.Linq; using System.Text; u

Head First设计模式之解释器模式

一.定义 给定一个语言,定义它的文法表示,并定义一个解释器,这个解释器使用该标识来解释语言中的句子. 主要解决:对于一些固定文法构建一个解释句子的解释器. 何时使用:如果一种特定类型的问题发生的频率足够高,那么可能就值得将该问题的各个实例表述为一个简单语言中的句子.这样就可以构建一个解释器,该解释器通过解释这些句子来解决该问题. 如何解决:构件语法树,定义终结符与非终结符. 二.结构 组成: AbstractExpression(抽象表达式):定义解释器的接口,约定解释器的解释操作. Termi

php设计模式之解释器模式

解释器设计模式用于分析一个实体的关键元素,并且针对每个元素都提供自己的解释或相应的动作. <?php /** * 解释器模式 */ class User { protected $_username; public function __construct($username) { $this->_username = $username; } public function getProfilePage() { $profile = "<h2>I like never

【GOF23设计模式】解释器模式 &amp; 访问者模式

来源:http://www.bjsxt.com/ 一.[GOF23设计模式]_解释器模式.访问者模式.数学表达式动态解析库式 1.解释器模式Interpreter  2.访问者模式Visitor 

【Unity与23种设计模式】解释器模式(Interpreter)

GoF中定义: "定义一个程序设计语言所需要的语句,并提供解释来解析(执行)该语言." 传统上,执行程序代码通常通过两种方式 第一种:编译程序 第二种:解释器 常见的使用解释器的程序设计语言 包含流行与网页设计领域中的脚本语言 如JavaScript.PHP.Ruby等 这些程序代码经过一般文本编辑器编写完成后放入指定的位置 就可以由应用程序中的解释器直接执行 包括Lua Unity中 编写好的脚本程序执行之前会被UnityEngine编译过 严格来说不算是解释器模式 但与十几年前的开

设计模式之解释器模式--- Pattern Interpreter

模式的定义 类型 行为类 模式的使用场景 优点 缺点 UML类图 角色介绍 模式的通用源码 输出结果 Android源码中的模式实现 杂谈 参考资料 (1).设计模式之禅-第27章 解释器模式 (2)解释器模式 https://github.com/simple-android-framework/android_design_patterns_analysis