解释器模式

四则运算问题
通过解释器模式来实现四则运算,如计算a+b-c的值,具体要求
1) 先输入表达式的形式,比如 a+b+c-d+e, 要求表达式的字母不能重复
2)在分别输入a ,b, c, d, e 的值
3) 最后求出结果:如图
传统方案解决四则运算问题分析
1) 编写一个方法,接收表达式的形式,然后根据用户输入的数值进行解析,得到结果
2) 问题分析:如果加入新的运算符,比如 * / ( 等等,不利于扩展,另外让一个方法来解析会造成程序结构混乱,不够清晰.
3) 解决方案:可以考虑使用解释器模式, 即: 表达式 -> 解释器(可以有多种) -> 结果
 
基本介绍
1) 在编译原理中,一个算术表达式通过词法分析器形成词法单元,而后这些词法单元再通过语法分析器构建语法分析树,最终形成一颗抽象的语法分析树。这里的词法分析器和语法分析器都可以看做是解释器
2) 解释器模式(Interpreter Pattern):是指给定一个语言(表达式),定义它的文法的一种表示,并定义一个解释器,使用该解释器来解释语言中的句子(表达式)
3) 应用场景
• 应用可以将一个需要解释执行的语言中的句子表示为一个抽象语法树
• 一些重复出现的问题可以用一种简单的语言来表达
• 一个简单语法需要解释的场景
4) 这样的例子还有,比如编译器、运算表达式计算、正则表达式、机器人等
原理类图
原理类图的说明-即(解释器模式的角色及职责)
1) Context: 是环境角色,含有解释器之外的全局信息.
2) AbstractExpression: 抽象表达式, 声明一个抽象的解释操作,这个方法为抽象语法树中所有的节点所共享
3) TerminalExpression: 为终结符表达式, 实现与文法中的终结符相关的解释操作
4) NonTermialExpression: 为非终结符表达式,为文法中的非终结符实现解释操作.
5) 说明: 输入Context he TerminalExpression 信息通过Client 输入即可
解释器模式来实现四则
1) 应用实例要求
通过解释器模式来实现四则运算,如计算a+b-c的值
2) 思路分析和图解(类图)
3) 代码实现
抽象类表达式:Expression
/**
 * 抽象类表达式,通过HashMap 键值对, 可以获取到变量的值*/
public abstract class Expression {
    // a + b - c
    // 解释公式和数值, key 就是公式(表达式) 参数[a,b,c], value就是就是具体值
    // HashMap {a=10, b=20}
    public abstract int interpreter(HashMap<String, Integer> var);
}

变量的解释器:VarExpression

/**
 * 变量的解释器*/
public class VarExpression extends Expression {
    private String key; // key=a,key=b,key=c
    public VarExpression(String key) {
        this.key = key;
    }
    // var 就是{a=10, b=20}
    // interpreter 根据 变量名称,返回对应值
    @Override
    public int interpreter(HashMap<String, Integer> var) {
        return var.get(this.key);
    }
}

SymbolExpression

/**
 * 抽象运算符号解析器 这里,每个运算符号,都只和自己左右两个数字有关系,
 * 但左右两个数字有可能也是一个解析的结果,无论何种类型,都是Expression类的实现类*/
public class SymbolExpression extends Expression {
    protected Expression left;
    protected Expression right;
    public SymbolExpression(Expression left, Expression right) {
        this.left = left;
        this.right = right;
    }
    //因为 SymbolExpression 是让其子类来实现,因此 interpreter 是一个默认实现
    @Override
    public int interpreter(HashMap<String, Integer> var) {
        // TODO Auto-generated method stub
        return 0;
    }
}

加法解释器

/**
 * 加法解释器*/
public class AddExpression extends SymbolExpression  {
    public AddExpression(Expression left, Expression right) {
        super(left, right);
    }
    //处理相加
    //var 仍然是 {a=10,b=20}..
    //一会debug 源码,就ok
    public int interpreter(HashMap<String, Integer> var) {
        //super.left.interpreter(var) : 返回 left 表达式对应的值 a = 10
        //super.right.interpreter(var): 返回right 表达式对应值 b = 20
        return super.left.interpreter(var) + super.right.interpreter(var);
    }
}

减法解释器

public class SubExpression extends SymbolExpression {
    public SubExpression(Expression left, Expression right) {
        super(left, right);
    }
    //求出left 和 right 表达式相减后的结果
    public int interpreter(HashMap<String, Integer> var) {
        return super.left.interpreter(var) - super.right.interpreter(var);
    }
}

计算器

public class Calculator {
    // 定义表达式
    private Expression expression;
    // 构造函数传参,并解析
    public Calculator(String expStr) { // expStr = a+b
        // 安排运算先后顺序
        Stack<Expression> stack = new Stack<>();
        // 表达式拆分成字符数组 
        char[] charArray = expStr.toCharArray();// [a, +, b]
        Expression left = null;
        Expression right = null;
        //遍历我们的字符数组, 即遍历  [a, +, b]
        //针对不同的情况,做处理
        for (int i = 0; i < charArray.length; i++) {
            switch (charArray[i]) {
            case '+': //
                left = stack.pop();// 从stack取出left => "a"
                right = new VarExpression(String.valueOf(charArray[++i]));// 取出右表达式 "b"
                stack.push(new AddExpression(left, right));// 然后根据得到left 和 right 构建 AddExpresson加入stack
                break;
            case '-': // 
                left = stack.pop();
                right = new VarExpression(String.valueOf(charArray[++i]));
                stack.push(new SubExpression(left, right));
                break;
            default: 
                //如果是一个 Var 就创建要给 VarExpression 对象,并push到 stack
                stack.push(new VarExpression(String.valueOf(charArray[i])));
                break;
            }
        }
        //当遍历完整个 charArray 数组后,stack 就得到最后Expression
        this.expression = stack.pop();
    }
    public int run(HashMap<String, Integer> var) {
        //最后将表达式a+b和 var = {a=10,b=20}
        //然后传递给expression的interpreter进行解释执行
        return this.expression.interpreter(var);
    }
}

测试调用

public class ClientTest {
    public static void main(String[] args) throws IOException {
        // TODO Auto-generated method stub
        String expStr = getExpStr(); // a+b
        HashMap<String, Integer> var = getValue(expStr);// var {a=10, b=20}
        Calculator calculator = new Calculator(expStr);
        System.out.println("运算结果:" + expStr + "=" + calculator.run(var));
    }
    // 获得表达式
    public static String getExpStr() throws IOException {
        System.out.print("请输入表达式:");
        return (new BufferedReader(new InputStreamReader(System.in))).readLine();
    }
    // 获得值映射
    public static HashMap<String, Integer> getValue(String expStr) throws IOException {
        HashMap<String, Integer> map = new HashMap<>();

        for (char ch : expStr.toCharArray()) {
            if (ch != '+' && ch != '-') {
                if (!map.containsKey(String.valueOf(ch))) {
                    System.out.print("请输入" + String.valueOf(ch) + "的值:");
                    String in = (new BufferedReader(new InputStreamReader(System.in))).readLine();
                    map.put(String.valueOf(ch), Integer.valueOf(in));
                }
            }
        }
        return map;
    }
}

 

解释器模式在Spring框架应用的源码剖析
1) Spring框架中 SpelExpressionParser就使用到解释器模式
2) 代码分析+Debug源码
先看这段代码

 结果肯定是96,它是怎么运转的,肯定是调用的getValue,我们先来看一下Expression,它是个接口,有几个实现类,那么这些实现类也肯定把接口里的方法实现了

 不过它是要先获得一个parser 对象,所以我们先来看SpelExpressionParser,它继承了TemplateAwareExpressionParser

 我们再往下看这个TemplateAwareExpressionParser ,它继承了ExpressionParser

 那我们再来看这个ExpressionParser 。。。发现它是个接口

 这下就知道关系了,然后我们再来看SpelExpressionParser ,它下面没有parseExpression 方法

 所以我们来看它的父类,确实有parseExpression  方法

parseExpression  这个方法看以看到,根据不同的情况,返回不同的Exception

 我们来看这个doParseExpression 方法,发现是个抽象的

 那么这个doParseExpression  应该被它的子类实现了,来看它的子类SpelExpressionParser ,果然有doParseExpression  方法

 然后这个重写的doParseExpression  方法里new InternalSpelExpressionParser 然后用的doParseExpression,那我来看这个doParseExpression

 它返回的是SpelExpression

这下就知道了,先new SpelExpressionParser,然后通过parser 对象的parseExpression 方法

3) 说明
 Expression 接口 表达式接口,下面有不同的实现类,比如SpelExpression, 或者CompositeStringExpression。使用时候,根据你创建的不同的Parser 对象,返回不同的 Expression 对象 ,使用得当 Expression对象,调用getValue 解释执行 表达式,最后得到结果
解释器模式的注意事项和细节
1) 当有一个语言需要解释执行,可将该语言中的句子表示为一个抽象语法树,就可以考虑使用解释器模式,让程序具有良好的扩展性
2) 应用场景:编译器、运算表达式计算、正则表达式、机器人等
3) 使用解释器可能带来的问题:解释器模式会引起类膨胀、解释器模式采用递归调用方法,将会导致调试非常复杂、效率可能降低.
 
 解释器模式相比较有一点难度,不要因为学不懂懊恼哈,开发中用的比较少
posted @ 2020-12-22 15:14  aBiu--  阅读(124)  评论(0编辑  收藏  举报