设计模式系统回顾(19)解释器模式

按照规定的语法(文法)来进行解析的一种设计模式
终结符表达式和非终结符表达式

  • 终结符表达式(Terminal Expression):实现文法中与终结符有关的解释操作。文法中每一个终结符都有一个具体的终结符表达式与之相对应。比如我们的 R=M+N 运算,M 和 N 就是终结符,对应的解析 M 和 N 的解释器就是终结符表达式
  • 非终符结表达式(Nonterminal Expression):实现文法中与非终结符有关的解释操作。文法中的每一条规则都对应了一个非终结符表达式。非终结表达式一般是文法中的运算符或者关键字,如上面公示:R=M+N 中的“+”号就是非终结符,解析“+”号的解释器就是一个非终结符表达式
public interface IExpression {
    int interpret();//用来解释表达式的方法
}

/**
 * 非终结表达式-抽象表达式
 */
public abstract class AbstractNonTerminalExpression implements IExpression{
    protected IExpression leftExpression;//非终结表达式左边表达式,一般为数字
    protected IExpression rightExpression;//非终结表达式右边表达式,一般为数字

    public AbstractNonTerminalExpression(IExpression leftExpression, IExpression rightExpression) {
        this.leftExpression = leftExpression;
        this.rightExpression = rightExpression;
    }
}

/**
 * 非终结表达式-具体表达式-减法表达式
 */
public class SubExpression extends AbstractNonTerminalExpression {

    public SubExpression(IExpression leftExpression, IExpression rightExpression) {
        super(leftExpression, rightExpression);
    }

    @Override
    public int interpret() {//解释器,将左右两个终结符的值相减
        return this.leftExpression.interpret() - this.rightExpression.interpret();
    }
}

public class NumberExpression implements IExpression{
    private int value;//终结表达式的值

    public NumberExpression(String value) {
        this.value = Integer.valueOf(value);
    }

    @Override
    public int interpret() {//解释非终结表达式
        return this.value;//直接返回值就行了
    }
}


public class ExpressionContext {
    private Stack<IExpression> stack = new Stack<>();//定义一个栈,计算一般都用栈,利用其后入先出的特性

    public ExpressionContext(String expression) {
        this.parse(expression);
    }

    private void parse(String expression) {
        String[] elementArr = expression.split(" ");//为了简单,直接以空格来切割,所以测试的时候每个数字和符号之间都要有空格
        for (int i=0;i<elementArr.length;i++){
            String element = elementArr[i];
            if (element.equals("+")){//加法
                IExpression leftExpression = stack.pop();//栈内元素出栈
                IExpression rightExpression = new NumberExpression(elementArr[++i]);//取出+号后的下一个元素
                IExpression addExpression = new AddExpression(leftExpression,rightExpression);
                stack.push(new NumberExpression(addExpression.interpret() + ""));//将计算结果入栈
            }else if (element.equals("-")){//减法
                IExpression leftExpression = stack.pop();//栈内元素出栈
                IExpression rightExpression = new NumberExpression(elementArr[++i]);//取出-号后的下一个元素
                IExpression subExpression = new SubExpression(leftExpression,rightExpression);
                stack.push(new NumberExpression(subExpression.interpret() + ""));//将计算结果入栈
            }else{
                stack.push(new NumberExpression(element));//如果是数字则直接入栈
            }
        }
    }

    public int calcuate(){//运算结果
        return stack.pop().interpret();//经过前面解析,到这里stack内只会剩下唯一一个数字,即运算结果
    }
}


public class TestInterpreter {
    public static void main(String[] args) {
        ExpressionContext context = new ExpressionContext("666 + 888 - 777");//注意每个符号间要包含空格
        System.out.println(context.calcuate());//输出计算结果
        context = new ExpressionContext("123 - 456 + 11");
        System.out.println(context.calcuate());//输出计算结果
    }
}

使用场景
1.有需要重复解析的问题
2.需要解释一些语法
优点
扩展性比较强
缺点
1.规则复杂时会引起类膨胀
2.影响效率

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值