💎一站式轻松地调用各大LLM模型接口,支持GPT4、智谱、星火、月之暗面及文生图 广告
27.2 解释器模式的定义 解释器模式(Interpreter Pattern)是一种按照规定语法进行解析的方案,在现在项目中使用较少,其定义如下: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所示。 ![](https://box.kancloud.cn/2016-08-14_57b0036bbd72c.jpg) 图27-4 解释器模式通用类图 ● AbstractExpression——抽象解释器 具体的解释任务由各个实现类完成,具体的解释器分别由TerminalExpression和Non-terminalExpression完成。 ● TerminalExpression——终结符表达式 实现与文法中的元素相关联的解释操作,通常一个解释器模式中只有一个终结符表达式,但有多个实例,对应不同的终结符。具体到我们例子就是VarExpression类,表达式中的每个终结符都在栈中产生了一个VarExpression对象。 ● NonterminalExpression——非终结符表达式 文法中的每条规则对应于一个非终结表达式,具体到我们的例子就是加减法规则分别对应到AddExpression和SubExpression两个类。非终结符表达式根据逻辑的复杂程度而增加,原则上每个文法规则都对应一个非终结符表达式。 ● Context——环境角色 具体到我们的例子中是采用HashMap代替。 解释器是一个比较少用的模式,以下为其通用源码,可以作为参考。抽象表达式通常只有一个方法,如代码清单27-8所示。 代码清单27-8 抽象表达式 public abstract class Expression {      //每个表达式必须有一个解析任务      public abstract Object interpreter(Context  ctx); } 抽象表达式是生成语法集合(也叫做语法树)的关键,每个语法集合完成指定语法解析任务,它是通过递归调用的方式,最终由最小的语法单元进行解析完成。终结符表达式如代码清单27-9所示。 代码清单27-9 终结符表达式 public class TerminalExpression extends Expression {      //通常终结符表达式只有一个,但是有多个对象      public Object interpreter(Context ctx) {              return null;      } } 通常,终结符表达式比较简单,主要是处理场景元素和数据的转换。 非终结符表达式如代码清单27-10所示。 代码清单27-10 非终结符表达式 public class NonterminalExpression extends Expression {      //每个非终结符表达式都会对其他表达式产生依赖      public NonterminalExpression(Expression... expression){      }            public Object interpreter(Context ctx) {              //进行文法处理              return null;      } } } 每个非终结符表达式都代表了一个文法规则,并且每个文法规则都只关心自己周边的文法规则的结果(注意是结果),因此这就产生了每个非终结符表达式调用自己周边的非终结符表达式,然后最终、最小的文法规则就是终结符表达式,终结符表达式的概念就是如此,不能够再参与比自己更小的文法运算了。 客户类如代码清单27-11所示。 代码清单27-11 客户类 public class Client {      public static void main(String[] args) {           Context ctx = new Context();           //通常定一个语法容器,容纳一个具体的表达式,通常为ListArray、LinkedList、Stack等类型           Stack&Expression> stack = null;            for(;;){                //进行语法判断,并产生递归调用           }                     //产生一个完整的语法树,由各个具体的语法分析进行解析           Expression exp = stack.pop();           //具体元素进入场景           exp.interpreter(ctx);      } } 通常Client是一个封装类,封装的结果就是传递进来一个规范语法文件,解析器分析后产生结果并返回,避免了调用者与语法解析器的耦合关系。