zoukankan      html  css  js  c++  java
  • 设计模式之解释器模式

       解释器模式(Interpreter Pattern)是一种按照规定语法进行解析的方案,在现在项目中使用较少(谁没事干会去写一个PHP或者Ruby的解析器),其定义如下:Given a language, define a representation for its grammar along with an interpreter that uses the representation to interpret sentences in the language。给定一个语言定义它的文法的一种表示,并定义一个解释器,该解释器使用该表示来解释语言中的句子

          解释器模式的通用类图如图27-4所示。

    • AbstractExpression 抽象解释器

          具体的解释任务由各个实现类完成,具体的解释器分别由TerminalExpression和NonterminalExpression完成。

    • TerminalExpression终结符表达式

          实现与文法中的元素相关联的解释操作,通常一个解释器模式中只有一个终结符表达式,但有多个实例,对应不同的终结符。具体到我们例子就是VarExpression类,表达式中的每个终结符都在堆栈中产生了一个VarExpression对象。

    • NonterminalExpression 非终结符表达式

          文法中的每条规则对应于一个非终结表达式,具体到我们的例子就是加减法规则分别对应到AddExpression和SubExpression两个类。非终结符表达式根据逻辑的复杂程度而增加,原则上每个文法规则都对应一个非终结符表达式。

    • Context 环境角色

    如下面的表达式:

    3 * 4 / 2 % 4

    可以使用如下文法来定义:

    expression::=value|symbol;

    symbol::=expression'+"expression|expression'-'expression|expression'%'expresson;

    value::=an integer;//一个整数值

    上面包含3条语法规则,第一句表示表达式的组成方式,其中value和symbol是后面2个语法单位的定义,每一条语句所定义的字符串如symbol和value称为语法构造成分或语法单位,符合::=表示“定义为”。

    语法单位对应终结符表达式和非终结符表达式,如本例symbol是非终结符表达式,他的组成元素可以使表达式,可以进一步分解,而value是终结符表达式,他的组成元素是基本的语法单位,不能再分解。

    每个文法规则都可以表示为一个由这些类的实例构成的抽象语法树。

    在解释器模式中,每一种终结符和非终结符都有一个具体类与之对应,正因为使用类来表示每一个语法规则,使得系统有较好的扩展性和灵活性

    实现  3 * 4 / 2 % 4 运算:

    1.抽象表达式类Node

    public interface Node {
    public int interpret();

    }

    2.终结符表达式类ValueNode(值节点类)

    public class ValueNode  implements Node{
        private int value;
        public ValueNode(int value)
        {
            this.value=value;
            
        }
        public int interpret()
        {
            return this.value;
        }
    }

    3.抽象非终结符表达式类SymbolNode

    public abstract class SymbolNode implements Node{
        protected Node left,right;
        public SymbolNode(Node left,Node right)
        {
            this.left=left;
            this.right=right;
        }
    
    }

    他包含了所有非终结符表达式的共有的数据和行为,在本例中,由于所有的非终结符都对应左右2个操作部分,因此在该类定义了left和right2个Node类型的对象,

    4.非终结符表达式MulNode

    public class MulNode extends SymbolNode{
        public MulNode(Node left,Node right)
        {
            super(left,right);
            
        }
        public int interpret()
        {
            return super.left.interpret()*super.right.interpret();
        }
    
    }

    5.非终结符表达式ModNode

    public class ModNode extends SymbolNode{
        public ModNode(Node left,Node right)
        {
            super(left,right);
            
        }
        public int interpret()
        {
            return super.left.interpret()%super.right.interpret();
        }
    }

    6:DivNode

    public class DivNode extends SymbolNode{
        public DivNode(Node left,Node right)
        {
            super(left,right);
            
        }
        public int interpret()
        {
            return super.left.interpret()/super.right.interpret();
        }
    
    }

    辅助代码:

    1.解释器封装类Calculator(计算器类)

    public class Calculator {
        private String statement;
        private Node node;
        public void bulid(String statement)
        {
            Node left=null,right=null;
            Stack stack=new Stack();
            
            String[] statementArr=statement.split(" ");
            for(int i=0;i<statementArr.length;i++)
            {
                if(statementArr[i].equalsIgnoreCase("*"))
                {
                    left=(Node)stack.pop();
                    int val=Integer.parseInt(statementArr[++i]);
                    right=new ValueNode(val);
                    stack.push(new MulNode(left,right));
                }
                else if(statementArr[i].equalsIgnoreCase("/"))
                {
                    left=(Node)stack.pop();
                    int val=Integer.parseInt(statementArr[++i]);
                    right=new ValueNode(val);
                    stack.push(new DivNode(left,right));
                  
                }
                else if(statementArr[i].equalsIgnoreCase("%"))
                {
                    left=(Node)stack.pop();
                    int val=Integer.parseInt(statementArr[++i]);
                    right=new ValueNode(val);
                    stack.push(new ModNode(left,right));
                  
                }
                else
                {
                    stack.push(new ValueNode(Integer.parseInt(statementArr[i])));
                }
                 
                
            }
            this.node=(Node)stack.pop();
    }
            public int compute()
            {
                return node.interpret();
            }
        
    
    }

    Calculator是本例核心之一,他极大地简化了客户类代码。在Calculator类中定义了如何构造一颗抽象的语法树,在构造了使用了stack。通过一系列操作,放置在栈中的是一个完整的表达式,通过栈的pop方法将其取出,再在compute()方法中调用该表达式的interpret方法,程序执行时将递归调用每一个子表达式的interpret方法,即执行每一个封装在终结符表达式类和非终结符表达式类中的interpret()方法。

    客户端测试类Client:

    public class Client {
        public static void main(String[ ] args)
        {
            String statement="3 * 4 / 2 % 4";
            Calculator c=new Calculator();
            c.bulid(statement);
            int result=c.compute();
            System.out.println(statement+"="+result);
            
        }
    
    }
  • 相关阅读:
    linux下遍历目录
    C++忽略字符大小写比较
    apue.h文件找不到的解决办法
    使用Django框架
    输出程序运行的时间(精确到微秒)
    好玩的代码(1)
    在日志文件中输出当前时间
    windows下安装storm1.1.0并启动
    eclipse配置hadoop2.7.2开发环境并本地跑起来
    windows下安装并启动hadoop2.7.2
  • 原文地址:https://www.cnblogs.com/youxin/p/2958120.html
Copyright © 2011-2022 走看看