interpreter(解释器模式)

一、引子
其实没有什么好的例子引入解释器模式,因为它描述了如何构成一个简单的语言解释器,主要应用在使用面向对象语言开发编译器中;在实际应用中,我们可能很少碰到去构造一个语言的文法的情况。
       虽然你几乎用不到这个模式,但是看一看还是能受到一定的启发的。

二、定义与结构
解释器模式的定义如下:定义语言的文法,并且建立一个解释器来解释该语言中的句子。它属于类的行为模式。这里的语言意思是使用规定格式和语法的代码。
       在GOF的书中指出:如果一种特定类型的问题发生的频率足够高,那么可能就值得将该问题的各个实例表述为一个简单语言中的句子。这样就可以构建一个解释器,该解释器通过解释这些句子来解决该问题。而且当文法简单、效率不是关键问题的时候效果最好。
       呵呵,这也就是解释器模式应用的环境了。
       让我们来看看神秘的解释器模式是由什么来组成的吧。
1)        抽象表达式角色:声明一个抽象的解释操作,这个接口为所有具体表达式角色(抽象语法树中的节点)都要实现的。
什么叫做抽象语法树呢?《Java与模式》中给的解释为:抽象语法树的每一个节点都代表一个语句,而在每个节点上都可以执行解释方法。这个解释方法的执行就代表这个语句被解释。由于每一个语句都代表这个语句被解释。由于每一个语句都代表一个常见的问题的实例,因此每一个节点上的解释操作都代表对一个问题实例的解答。
2)        终结符表达式角色:具体表达式。
a)        实现与文法中的终结符相关联的解释操作
b)        而且句子中的每个终结符需要该类的一个实例与之对应
3)        非终结符表达式角色:具体表达式。
a)        文法中的每条规则R::=R1R2…Rn都需要一个非终结符表带式角色
b)        对于从R1到Rn的每个符号都维护一个抽象表达式角色的实例变量
c)        实现解释操作,解释一般要递归地调用表示从R1到Rn的那些对象的解释操作
4)        上下文(环境)角色:包含解释器之外的一些全局信息。
5)        客户角色:
a)        构建(或者被给定)表示该文法定义的语言中的一个特定的句子的抽象语法树
b)        调用解释操作

放上张解释器结构类图吧,这也是来自于GOF的书中。

呵呵,对每一个角色都给出了详细的职责,而且在类图中给出五个角色之间的关系。这样实现起来也不是很困难了,下面举了一个简单的例子,希望能加深你对解释器模式的理解。

三、举例
来举一个加减乘除的例子吧,实现思路来自于《java与模式》中的例子。每个角色的功能按照上面提到的规范来实现。

[java] view plain copy
//上下文(环境)角色,使用HashMap来存储变量对应的数值  

class Context  

{  

       private Map valueMap = new HashMap();  

       public void addValue(Variable x , int y)  

       {  

              Integer yi = new Integer(y);  

              valueMap.put(x , yi);  

       }  

       public int LookupValue(Variable x)  

       {  

              int i = ((Integer)valueMap.get(x)).intValue();  

              return i ;  

       }  

}  

[java] view plain copy
//抽象表达式角色,也可以用接口来实现  

abstract class Expression  

{  

       public abstract int interpret(Context con);  

}  

[java] view plain copy
//终结符表达式角色  

class Constant extends Expression  

{  

       private int i ;  

       public Constant(int i)  

       {  

              this.i = i;  

       }  

       public int interpret(Context con)  

       {  

              return i ;  

       }  

}  

[java] view plain copy
class Variable extends Expression  

{  

       public int interpret(Context con)  

       {  

              //this为调用interpret方法的Variable对象  

              return con.LookupValue(this);  

       }  

}  

[java] view plain copy
//非终结符表达式角色  

class Add extends Expression  

{  

       private Expression left ,right ;  

       public Add(Expression left , Expression right)  

       {  

              this.left = left ;  

              this.right= right ;  

       }  

       public int interpret(Context con)  

       {  

              return left.interpret(con) + right.interpret(con);  

       }  

}  

[java] view plain copy
class Subtract extends Expression  

{  

       private Expression left , right ;  

       public Subtract(Expression left , Expression right)  

       {  

              this.left = left ;  

              this.right= right ;  

       }  

       public int interpret(Context con)  

       {  

              return left.interpret(con) - right.interpret(con);  

       }  

}  

[java] view plain copy
class Multiply extends Expression  

{  

       private Expression left , right ;  

       public Multiply(Expression left , Expression right)  

       {  

              this.left = left ;  

              this.right= right ;  

       }  

       public int interpret(Context con)  

       {  

              return left.interpret(con) * right.interpret(con);  

       }  

}  

[java] view plain copy
class Division extends Expression  

{  

       private Expression left , right ;  

       public Division(Expression left , Expression right)  

       {  

              this.left = left ;  

              this.right= right ;  

       }  

       public int interpret(Context con)  

       {  

              try{  

                            return left.interpret(con) / right.interpret(con);  

              }catch(ArithmeticException ae)  

              {  

                     System.out.println("被除数为0!");  

                     return -11111;  

              }  

       }  

}  

[java] view plain copy
//测试程序,计算 (a*b)/(a-b+2)  

public class Test  

{  

       private static Expression ex ;  

       private static Context con ;  

       public static void main(String[] args)  

       {  

              con = new Context();  

              //设置变量、常量  

              Variable a = new Variable();  

              Variable b = new Variable();  

              Constant c = new Constant(2);  

//为变量赋值  

              con.addValue(a , 5);  

              con.addValue(b , 7);  

//运算,对句子的结构由我们自己来分析,构造  

              ex = new Division(new Multiply(a , b), new Add(new Subtract(a , b) , c));  

              System.out.println("运算结果为:"+ex.interpret(con));  

       }  

}  

解释器模式并没有说明如何创建一个抽象语法树,因此它的实现可以多种多样,在上面我们是直接在Test中提供的,当然还有更好、更专业的实现方式。
对于终结符,GOF建议采用享元模式来共享它们的拷贝,因为它们要多次重复出现。但是考虑到享元模式的使用局限性,我建议还是当你的系统中终结符重复的足够多的时候再考虑享元模式(关于享元模式,请参考我的《深入浅出享元模式》)。

四、优缺点
解释器模式提供了一个简单的方式来执行语法,而且容易修改或者扩展语法。一般系统中很多类使用相似的语法,可以使用一个解释器来代替为每一个规则实现一个解释器。而且在解释器中不同的规则是由不同的类来实现的,这样使得添加一个新的语法规则变得简单。
但是解释器模式对于复杂文法难以维护。可以想象一下,每一个规则要对应一个处理类,而且这些类还要递归调用抽象表达式角色,多如乱麻的类交织在一起是多么恐怖的一件事啊!

五、总结
这样对解释器模式应该有了些大体的认识了吧,由于这个模式使用的案例匮乏,所以本文大部分观点直接来自于GOF的原著。只是实例代码是亲自实现并调试通过的。

由于解释器模式基本不用,我就直接转载网上的了,大家看看涨涨姿势http://blog.csdn.net/ylchou/article/details/7594135

一、引子

其实没有什么好的例子引入解释器模式,因为它描述了如何构成一个简单的语言解释器,主要应用在使用面向对象语言开发编译器中;在实际应用中,我们可能很少碰到去构造一个语言的文法的情况。

虽然你几乎用不到这个模式,但是看一看还是能受到一定的启发的。

二、定义与结构

解释器模式的定义如下:定义语言的文法,并且建立一个解释器来解释该语言中的句子。它属于类的行为模式。这里的语言意思是使用规定格式和语法的代码。

在GOF的书中指出:如果一种特定类型的问题发生的频率足够高,那么可能就值得将该问题的各个实例表述为一个简单语言中的句子。这样就可以构建一个解释器,该解释器通过解释这些句子来解决该问题。而且当文法简单、效率不是关键问题的时候效果最好。

呵呵,这也就是解释器模式应用的环境了。

让我们来看看神秘的解释器模式是由什么来组成的吧。

1)        抽象表达式角色:声明一个抽象的解释操作,这个接口为所有具体表达式角色(抽象语法树中的节点)都要实现的。

什么叫做抽象语法树呢?《Java与模式》中给的解释为:抽象语法树的每一个节点都代表一个语句,而在每个节点上都可以执行解释方法。这个解释方法的执行就代表这个语句被解释。由于每一个语句都代表这个语句被解释。由于每一个语句都代表一个常见的问题的实例,因此每一个节点上的解释操作都代表对一个问题实例的解答。

2)        终结符表达式角色:具体表达式。

a)        实现与文法中的终结符相关联的解释操作

b)        而且句子中的每个终结符需要该类的一个实例与之对应

3)        非终结符表达式角色:具体表达式。

a)        文法中的每条规则R::=R1R2…Rn都需要一个非终结符表带式角色

b)        对于从R1到Rn的每个符号都维护一个抽象表达式角色的实例变量

c)        实现解释操作,解释一般要递归地调用表示从R1到Rn的那些对象的解释操作

4)        上下文(环境)角色:包含解释器之外的一些全局信息。

5)        客户角色:

a)        构建(或者被给定)表示该文法定义的语言中的一个特定的句子的抽象语法树

b)        调用解释操作

放上张解释器结构类图吧,这也是来自于GOF的书中。

呵呵,对每一个角色都给出了详细的职责,而且在类图中给出五个角色之间的关系。这样实现起来也不是很困难了,下面举了一个简单的例子,希望能加深你对解释器模式的理解。

三、举例

来举一个加减乘除的例子吧,实现思路来自于《java与模式》中的例子。每个角色的功能按照上面提到的规范来实现。

[java] view plain copy

  1. //上下文(环境)角色,使用HashMap来存储变量对应的数值
  2. class Context
  3. {
  4. private Map valueMap = new HashMap();
  5. public void addValue(Variable x , int y)
  6. {
  7. Integer yi = new Integer(y);
  8. valueMap.put(x , yi);
  9. }
  10. public int LookupValue(Variable x)
  11. {
  12. int i = ((Integer)valueMap.get(x)).intValue();
  13. return i ;
  14. }
  15. }

[java] view plain copy

  1. //抽象表达式角色,也可以用接口来实现
  2. abstract class Expression
  3. {
  4. public abstract int interpret(Context con);
  5. }

[java] view plain copy

  1. //终结符表达式角色
  2. class Constant extends Expression
  3. {
  4. private int i ;
  5. public Constant(int i)
  6. {
  7. this.i = i;
  8. }
  9. public int interpret(Context con)
  10. {
  11. return i ;
  12. }
  13. }

[java] view plain copy

  1. class Variable extends Expression
  2. {
  3. public int interpret(Context con)
  4. {
  5. //this为调用interpret方法的Variable对象
  6. return con.LookupValue(this);
  7. }
  8. }

[java] view plain copy

  1. //非终结符表达式角色
  2. class Add extends Expression
  3. {
  4. private Expression left ,right ;
  5. public Add(Expression left , Expression right)
  6. {
  7. this.left = left ;
  8. this.right= right ;
  9. }
  10. public int interpret(Context con)
  11. {
  12. return left.interpret(con) + right.interpret(con);
  13. }
  14. }

[java] view plain copy

  1. class Subtract extends Expression
  2. {
  3. private Expression left , right ;
  4. public Subtract(Expression left , Expression right)
  5. {
  6. this.left = left ;
  7. this.right= right ;
  8. }
  9. public int interpret(Context con)
  10. {
  11. return left.interpret(con) - right.interpret(con);
  12. }
  13. }

[java] view plain copy

  1. class Multiply extends Expression
  2. {
  3. private Expression left , right ;
  4. public Multiply(Expression left , Expression right)
  5. {
  6. this.left = left ;
  7. this.right= right ;
  8. }
  9. public int interpret(Context con)
  10. {
  11. return left.interpret(con) * right.interpret(con);
  12. }
  13. }

[java] view plain copy

  1. class Division extends Expression
  2. {
  3. private Expression left , right ;
  4. public Division(Expression left , Expression right)
  5. {
  6. this.left = left ;
  7. this.right= right ;
  8. }
  9. public int interpret(Context con)
  10. {
  11. try{
  12. return left.interpret(con) / right.interpret(con);
  13. }catch(ArithmeticException ae)
  14. {
  15. System.out.println("被除数为0!");
  16. return -11111;
  17. }
  18. }
  19. }

[java] view plain copy

  1. //测试程序,计算 (a*b)/(a-b+2)
  2. public class Test
  3. {
  4. private static Expression ex ;
  5. private static Context con ;
  6. public static void main(String[] args)
  7. {
  8. con = new Context();
  9. //设置变量、常量
  10. Variable a = new Variable();
  11. Variable b = new Variable();
  12. Constant c = new Constant(2);
  13. //为变量赋值
  14. con.addValue(a , 5);
  15. con.addValue(b , 7);
  16. //运算,对句子的结构由我们自己来分析,构造
  17. ex = new Division(new Multiply(a , b), new Add(new Subtract(a , b) , c));
  18. System.out.println("运算结果为:"+ex.interpret(con));
  19. }
  20. }

解释器模式并没有说明如何创建一个抽象语法树,因此它的实现可以多种多样,在上面我们是直接在Test中提供的,当然还有更好、更专业的实现方式。

对于终结符,GOF建议采用享元模式来共享它们的拷贝,因为它们要多次重复出现。但是考虑到享元模式的使用局限性,我建议还是当你的系统中终结符重复的足够多的时候再考虑享元模式(关于享元模式,请参考我的《深入浅出享元模式》)。

四、优缺点

解释器模式提供了一个简单的方式来执行语法,而且容易修改或者扩展语法。一般系统中很多类使用相似的语法,可以使用一个解释器来代替为每一个规则实现一个解释器。而且在解释器中不同的规则是由不同的类来实现的,这样使得添加一个新的语法规则变得简单。

但是解释器模式对于复杂文法难以维护。可以想象一下,每一个规则要对应一个处理类,而且这些类还要递归调用抽象表达式角色,多如乱麻的类交织在一起是多么恐怖的一件事啊!

五、总结

这样对解释器模式应该有了些大体的认识了吧,由于这个模式使用的案例匮乏,所以本文大部分观点直接来自于GOF的原著。只是实例代码是亲自实现并调试通过的。

时间: 2024-10-13 00:19:02

interpreter(解释器模式)的相关文章

(16):(行为型模式) Interpreter 解释器模式

(16):(行为型模式) Interpreter 解释器模式,布布扣,bubuko.com

设计模式15:Interpreter 解释器模式(行为型模式)

Interpreter 解释器模式(行为型模式) 动机(Motivation) 在软件构建过程中,如果某一特定领域的问题比较复杂,类似的模式不断重复出现,如果使用普通的编程方式来实现将面临非常频繁的变化. 在这种情况下,将特定领域的问题表达为某种语法规则下的句子,然后构建一个解释器来解释这样的句子,从而达到解决问题的目的. 意图(Intent) 给定一个语言,定义它的文法的一种表示,并定义一种解释器,这个解释器用来解释语言中的句子.——<设计模式>GoF 中文数字转换为阿拉伯数字 public

设计模式(15)--Interpreter(解释器模式)--行为型

1.模式定义: 解释器模式是类的行为模式.给定一个语言之后,解释器模式可以定义出其文法的一种表示,并同时提供一个解释器.客户端可以使用这个解释器来解释这个语言中的句子. 2.模式特点: 解释器模式在实际的系统开发中使用的非常少,因为它会引起效率.性能以及维护等问题,一般在大中型的框架型项目能够找到它的身影,比如一些数据分析工具.报表设计工具.科学计算工具等等,若你确实遇到“一种特定类型的问题发生的频率足够高”的情况,准备使用解释器模式时,可以考虑一下Expression4J.MESP(Math

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

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

设计模式(二十)解释器模式(Interpreter)-行为型

解释器模式Interpreter 解释器模式在软件开发中应用的比较少,它主要用在底层的编程语言设计上,因此不太容易理解. 联想:传一个算数表达式,对加减乘除自动匹配,能够自动计算其结果. 原理图 解释器模式实现原理图 单个运算符的数学公式计算机可以用来理解这个解释器模式的使用. 参考文献 [1] 郭峰.深入浅出设计莫模式[M].中国铁道出版社,2013(1):415-423.

[工作中的设计模式]解释器模式模式Interpreter

一.模式解析 解释器模式是类的行为模式.给定一个语言之后,解释器模式可以定义出其文法的一种表示,并同时提供一个解释器.客户端可以使用这个解释器来解释这个语言中的句子. 以上是解释器模式的类图,事实上我很少附上类图,但解释器模式确实比较抽象,为了便于理解还是放了上来,此模式的要点是: 1.客户端提供一个文本.表达式或者其他,约定解析格式 2.针对文本中可以分为终结符表达式和非终结符表达式, 3.终结符表达式无需进一步解析,但仍需要转化为抽象接口的实例 4.针对非终结表达式,没一种标示需要定义一种解

设计模式的征途—23.解释器(Interpreter)模式

虽然目前计算机编程语言有好几百种,但有时人们还是希望用一些简单的语言来实现特定的操作,只需要向计算机输入一个句子或文件,就能按照预定的文法规则来对句子或文件进行解释.例如,我们想要只输入一个加法/减法表达式,它就能够计算出表达式结果.例如输入“1+2+3-4+1”时,将输出计算结果为3.像C++,Java或C#都无法直接解释类似这样的字符串,因此用户必须自定义一套文法规则来实现对这些语句的解释,即设计一个自定义语言.如果所基于的编程语言是面向对象语言,此时可以使用解释器模式实现自定义语言. 解释

[设计模式-行为型]解释器模式(Interpreter)

一句话 看起来是用来解释一种语言的文法.(类似不同的解释器子类解释不同的字符) 和编译器类似的解释器, 实际状况可能使用的比较少. 概括 解析 INTERPRETER-俺有一个<泡MM真经>,上面有各种泡MM的攻略,比如说去吃西餐的步骤.去看电影的方法等等,跟MM约会时,只要做一个Interpreter,照着上面的脚本执行就可以了. 解释器模式:给定一个语言后,解释器模式可以定义出其文法的一种表示,并同时提供一个解释器.客户端可以使用这个解释器来解释这个语言中的句子.解释器模式将描述怎样在有了

&quot;围观&quot;设计模式(27)--行为型之解释器模式(Interpreter Pattern)

解析器是一种按照规定的语法进行解析的例子,在现在的项目中使用较少,定义如下:给定一门语言,定义它的文法的一种表示,并定义一个解释器,该解释器用于解释语言中的句子. 个人理解 解释器模式在项目中很少使用,因为他会引起效率.性能以及维护等问题,准备使用该模式时可以考虑开源框架如:Expression4J.MESP.Jep等.解释器模式一般用来解析比较标准的字符集,比如说SQL语法分析等. 解释器角色 解释器模式主要包含如下几个角色: AbstractExpression: 抽象表达式.声明一个抽象的