解释器模式

思考并回答以下问题:

本章导学

解释器模式用于描述如何构成一个简单的语言解释器,主要应用于使用面向对象语言开发的解释器的设计。当需要开发一个新的语言时,开发人员可以考虑使用解释器模式。在实际应用中,用户也许很少碰到去构造一个语言的情况,虽然很少使用,但是对它进行学习能够加深对面向对象思想的理解,并且掌握编程语言中语法规则解释的原理和过程。

本章将学习解释器模式的定义和结构,并结合实例学习如何使用解释器模式构造一个新的语言,以及如何通过终结符表达式和非终结符表达式在类中封装语言的文法规则。

本章知识点

  • 解释器模式的定义
  • 解释器模式的结构
  • 解释器模式的实现
  • 解释器模式的应用
  • 文法规则和抽象语法树
  • 解释器模式的优缺点
  • 解释器模式的适用环境

解释器模式概述

虽然目前计算机编程语言有几百种,但有时人们还是希望能用一些简单的语言来实现一些特定的操作,用户只要向计算机输入一个句子或文件,它就能够按照预先定义的文法规则对句子或文件进行解释,从而实现相应的功能。例如提供一个简单的加法/减法解释器,只要输入一个加法/减法表达式,它就能够计算出表达式结果,如图1所示,当输入字符串表达式为“1+2+3-4+1”时,将输出计算结果“3”。

图1 加法/减法解释器示意图

众所周知,像C#、C+和Java等语言无法直接解释类似“1+2+3-4+1”这样的字符串(如果直接作为数学表达式则可以解释),必须定义一套文法规则来实现对这些语句的解释,即设计一个自定义语言。在实际开发中,这些简单的自定义语言可以基于现有的编程语言来设计,如果所基于的编程语言是面向对象语言,则可以使用解释器模式来实现自定义语言。

解释器模式是一种使用频率相对较低但学习难度相对较大的设计模式,用于描述如何使用面向对象语言构成一个简单的语言解释器。在某些情况下,为了更好地描述某一些特定类型的问题,可以创建一种新的语言,这种语言拥有自己的表达式和结构,即文法规则,这些问题的实例将对应该语言中的句子。此时,用户可以使用解释器模式来设计这种新的语言。另外,对解释器模式进行学习能够加深用户对面向对象思想的理解,并且理解编程语言中文法规则的解释过程。

解释器模式的定义如下:

1
给定一个语言,定义它的文法的一种表示,并定义一个解释器,这个解释器使用该表示来解释语言中的句子。  

在解释器模式的定义中所指的“语言”是使用规定格式和语法的代码,解释器模式是一种类行为型模式。

文法规则和抽象语法树

解释器模式描述了如何为简单的语言定义一个文法,如何在该语言中表示一个句子,以及如何解释这些句子。在正式分析解释器模式结构之前,先来学习如何表示一个语言的文法规则以及如何构造一棵抽象语法树。

在前面所提到的加法/减法解释器中,每一个输入表达式,例如“1+2+3-4+1”,都包含了3个语言单位,可以使用以下文法规则来定义:

1
2
3
expression ::= value | operation
operation ::= expression '+' expression | expression '-' expression
value ::= an integer //一个整数值

该文法规则包含3条语句,第一条表示表达式的组成方式,其中,value和operation是后面两个语言单位的定义,每一条语句所定义的字符串(如operation和value)称为语言构造成分或语言单位,符号“::=”是“定义为”的意思,其左边的语言单位通过右边进行说明和定义,语言单位对应终结符表达式和非终结符表达式。例如本规则中的operation是非终结符表达式,它的组成元素仍然可以是表达式,还可以进一步分解,而value是终结符表达式,它的组成元素是最基本的语言单位,不能再进行分解。

在文法规则定义中可以使用一些符号表示不同的含义,例如使用“|”表示“或”,使用“{”和“}”表示“组合”,使用“*”表示出现“0次或多次”等。其中,使用频率最高的符号是表示“或”关系的“|”,例如文法规则“boolValue::=0|1”表示终结符表达式boolValue的取值可以为0或者1。

除了使用文法规则来定义一个语言外,在解释器模式中还可以通过一种被称为抽象语法树(Abstract Syntax Tree,AST)的图形方式来直观地表示语言的构成,每一棵抽象语法树对应一个语言实例,例如加法/减法解释器中的表达式“1+2+3-4+1”可以通过如图2所示的抽象语法树来表示。

图2 抽象语法树示意图

在该抽象语法树中,可以通过终结符表达式value和非终结符表达式operation组成复杂的语句,每个文法规则的语言实例都可以表示为一棵抽象语法树,即每一条具体的语句都可以用类似图2所示的抽象语法树来表示。在该图中,终结符表达式类的实例作为树的叶子结点,而非终结符表达式类的实例作为非叶子结点,它们可以将终结符表达式类的实例以及包含终结符和非终结符实例的子表达式作为其子结点。抽象语法树描述了如何构成一个复杂的句子,通过对抽象语法树的分析,可以识别出语言中的终结符类和非终结符类。

解释器模式的结构与实现

解释器模式的结构

由于表达式可以分为终结符表达式和非终结符表达式,所以解释器模式的结构与组合模式的结构有些类似,但在解释器模式中包含更多的组成元素,它的结构如图3所示。

图3 解释器模式结构图

由图3可知,解释器模式包含以下4个角色。

(1)AbstractExpression(抽象表达式):在抽象表达式中声明了抽象的解释操作,它是所有终结符表达式和非终结符表达式的公共父类。

(2)TerminalExpression(终结符表达式):终结符表达式是抽象表达式的子类,它实现了与文法中的终结符相关联的解释操作,在句子中的每一个终结符都是该类的一个实例。通常,在一个解释器模式中只有少数几个终结符表达式类,它们的实例可以通过非终结符表达式组成较为复杂的句子。

(3)NonterminalExpression(非终结符表达式):非终结符表达式也是抽象表达式的子类,它实现了文法中非终结符的解释操作,由于在非终结符表达式中可以包含终结符表达式,也可以继续包含非终结符表达式,因此其解释操作一般通过递归的方式来完成。

(4)Context(环境类):环境类又称为上下文类,它用于存储解释器之外的一些全局信息,通常临时存储了需要解释的语句。

解释器模式的实现

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

对于所有的终结符和非终结符,首先需要抽象出一个公共父类,即抽象表达式类。其典型代码如下:

1
2
3
4
abstract class AbstractExpression
{
public abstract void Interpret(Context ctx);
}

终结符表达式类和非终结符表达式类都是抽象表达式类的子类,对于终结符表达式类,其代码很简单,主要是对终结符元素进行处理。其典型代码如下:
1
2
3
4
5
6
7
class TerminalExpression : AbstractExpression 
{
public override void Interpret(Context ctx)
{
//终结符表达式的解释操作
}
}

对于非终结符表达式,其代码相对比较复杂,因为可以通过非终结符将表达式组合成更加复杂的结构,对于包含两个操作元素的非终结符表达式类,其典型代码如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
class NonterminalExpression : AbstractExpression 
{
private AbstractExpression left;
private AbstractExpression right;

public NonterminalExpression(AbstractExpression left,AbstractExpression right)
{
this.left=left;
this.right=right;
}

public override void Interpret(Context ctx)
{
//递归调用每一个组成部分的interpret()方法
//在递归调用时指定组成部分的连接方式,即非终结符的功能
}
}

除了用于表示表达式的类以外,通常在解释器模式中还提供了一个环境类Context,用于存储一些全局信息,在环境类中一般包含了一个Hashtable或List等类型的集合对象(也可以直接由Hashtable等集合类充当环境类),存储一系列公共信息,例如变量名与值的映射关系(key/value等),用于在执行具体的解释操作时从中获取相关信息。其典型代码片段如下:
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
class Context
{
private Hashtable ht = new Hashtable();

//往环境类中设值
public void Assign(string key, string value)
{
ht.Add(key,value);
}

//获取存储在环境类中的值
public string Lookup(string key)
{
return (string)ht[key];
}
}

环境类Context的对象通常作为参数被传递到所有表达式的解释方法Interpret()中,可以在环境类对象中存储和访问表达式解释器的状态,向表达式解释器提供一些全局的、公共的数据,此外,还可以在环境类中增加一些所有表达式解释器都共有的功能,以减轻解释器的职责。当系统无须提供全局公共信息时可以省略环境类,根据实际情况决定是否需要环境类。

解释器模式的应用实例

下面通过一个应用实例来进一步学习和理解解释器模式。

1.实例说明

某软件公司要开发一套机器人控制程序,在该机器人控制程序中包含一些简单的英文控制指令,每一个指令对应一个表达式(expression),该表达式可以是简单表达式也可以是复合表达式。每一个简单表达式由移动方向(direction)、移动方式(action)和移动距离(distance)三部分组成,其中,移动方向包括向上(up)、向下(down)、向左(left)、向右(right);移动方式包括移动(move)和快速移动(run);移动距离为一个正整数。两个表达式之间可以通过与(and)连接,形成复合(composite)表达式。

用户通过对图形化的设置界面进行操作可以创建一个机器人控制指令,机器人在收到指令后将按照指令的设置进行移动,例如输入控制指令“up move 5”将“向上移动5个单位”;输入控制指令“down run 10 and left move 20”将“向下快速移动10个单位再向左移动20个单位”。

现使用解释器模式来设计该程序并模拟实现。

2.实例类图

根据上述需求描述,用形式化语言来表示该简单语言的文法规则如下:

1
2
3
4
5
expression ::= direction action distance | composite //表达式
composite ::= expression 'and' expression //复合表达式
direction ::= 'up' | 'down' | 'left' | 'right' //移动方向
action ::= 'move' | 'run' //移动方式
distance ::= an integer //移动距离

该语言一共定义了5条文法规则,对应5个语言单位,这些语言单位可以分为两类,一类为终结符(也称为终结符表达式),例如direction,action和distance,它们是语言的最小组成单位,不能再进行拆分;另一类为非终结符(也称为非终结符表达式),例如expression和composite,它们都是一个完整的句子,包含一系列终结符或非终结符。

针对5条文法规则,分别提供5个类来实现,其中,终结符表达式direction,action和diatance对应DirectionNode类、ActionNode类和DistanceNode类,非终结符表达式expression和composite对应SentenceNode类和AndNode类。

可以通过抽象语法树来表示具有解释过程,例如机器人控制指令“down run 10 and left move 20”对应的抽象语法树如图4所示。

图4 机器人控制程序抽象语法树实例

机器人控制程序实例的基本结构如图5所示。

图5 机器人控制程序结构图

在图5中,AbstractNode充当抽象表达式角色,DirectionNode、ActionNode和DistanceNode充当终结符表达式角色,AndNode和SentenceNode充当非终结符表达式角色。

3.实例代码

(1)AbstractNode:抽象节点类,充当抽象表达式角色。

1
2
3
4
5
6
7
namespace InterpreterSample
{
abstract class AbstractNode
{
public abstract string Interpret();
}
}

(2)AndNode:And节点类,充当非终结符表达式角色。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
namespace InterpreterSample
{
class AndNode : AbstractNode
{
private AbstractNode left; //And的左表达式
private AbstractNode right; //And的右表达式

public AndNode(AbstractNode left, AbstractNode right)
{
this.left = left;
this.right = right;
}

//And表达式解释操作
public override string Interpret()
{
return left.Interpret() + "再" + right.Interpret();
}
}
}

(3)SentenceNode:简单句子节点类,充当非终结符表达式角色。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
namespace InterpreterSample
{
class SentenceNode : AbstractNode
{
private AbstractNode direction;
private AbstractNode action;
private AbstractNode distance;

public SentenceNode(AbstractNode direction,AbstractNode action,AbstractNode distance)
{
this.direction = direction;
this.action = action;
this.distance = distance;
}

//简单句子的解释操作
public override string Interpret()
{
return direction.Interpret() + action.Interpret() + distance.Interpret();
}
}
}

(4)DirectionNode:方向节点类,充当终结符表达式角色。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
namespace InterpreterSample
{
class DirectionNode : AbstractNode
{
private string direction;

public DirectionNode(string direction)
{
this.direction = direction;
}

//方向表达式的解释操作
public override string Interpret()
{
if (direction.Equals("up"))
{
return "向上";
}
else if (direction.Equals("down"))
{
return "向下";
}
else if (direction.Equals("left"))
{
return "向左";
}
else if (direction.Equals("right"))
{
return "向右";
}
else
{
return "无效指令";
}
}
}
}

(5)ActionNode:动作节点类,充当终结符表达式角色。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
namespace InterpreterSample
{
class ActionNode : AbstractNode
{
private string action;

public ActionNode(string action)
{
this.action = action;
}

//动作(移动方式)表达式的解释操作
public override string Interpret()
{
if (action.Equals("move"))
{
return "移动";
}
else if (action.Equals("run"))
{
return "快速移动";
}
else
{
return "无效指令";
}
}
}
}

(6)DistanceNode:距离节点类,充当终结符表达式角色。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
namespace InterpreterSample
{
class DistanceNode : AbstractNode
{
private string distance;

public DistanceNode(string distance)
{
this.distance = distance;
}

//距离表达式的解释操作
public override string Interpret()
{
return this.distance;
}
}
}

(7)InstructionHandler:指令处理类,工具类,提供相应的方法对输入指令进行处理。它将输入指令分隔为字符串数组,将第1个、第2个和第3个单词组合成一个句子,并存入栈中;如果发现有单词“and”,则将“and”后的第1个、第2个和第3个单词组合成一个新的句子作为“and”的右表达式,并从栈中取出原先所存句子作为左表达式,然后组合成一个And结点存入栈中。依此类推,直到整个指令解析结束。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
using System;
using System.Collections;

namespace InterpreterSample
{
class InstructionHandler
{
private AbstractNode node;

public void Handle(string instruction)
{
AbstractNode left = null, right = null;
AbstractNode direction = null, action = null, distance = null;
Stack stack = new Stack(); //声明一个栈对象用于存储抽象语法树
string[] words = instruction.Split(' '); //以空格分隔指令字符串

for (int i = 0; i < words.Length; i++)
{
//本实例采用栈的方式来处理指令,如果遇到“and”,则将其后的三个单词作为三个终结符表达式连成一个简单句子SentenceNode作为“and”的右表达式,而将从栈顶弹出的表达式作为“and”的左表达式,最后将新的“and”表达式压入栈中。
if (words[i].Equals("and"))
{
left = (AbstractNode)stack.Pop(); //弹出栈顶表达式作为左表达式
string word1= words[++i];
direction = new DirectionNode(word1);
string word2 = words[++i];
action = new ActionNode(word2);
string word3 = words[++i];
distance = new DistanceNode(word3);
right = new SentenceNode(direction,action,distance); //右表达式
stack.Push(new AndNode(left,right)); //将新表达式压入栈中
}
//如果是从头开始进行解释,则将前三个单词组成一个简单句子SentenceNode并将该句子压入栈中
else
{
string word1 = words[i];
direction = new DirectionNode(word1);
string word2 = words[++i];
action = new ActionNode(word2);
string word3 = words[++i];
distance = new DistanceNode(word3);
left = new SentenceNode(direction,action,distance);
stack.Push(left); //将新表达式压入栈中
}
}
this.node = (AbstractNode)stack.Pop(); //将全部表达式从栈中弹出
}

public string Output()
{
string result = node.Interpret(); //解释表达式
return result;
}
}
}

(8)Program:客户端测试类。

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
using System;

namespace InterpreterSample
{
class Program
{
static void Main(string[] args)
{
//string instruction = "down run 10 and left move 20";
string instruction = "up move 5 and down run 10 and left move 5";
InstructionHandler handler = new InstructionHandler();
handler.Handle(instruction);
string outString;
outString = handler.Output();
Console.WriteLine(outString);

Console.Read();
}
}
}

4.结果及分析

编译并运行程序,输出结果如下:

1
向下快速移动10再向左移动20

如果将输入指令改为“up move 5 and down run 10 and left move 5”,则输出结果如下:
1
向上移动5再向下快速移动10再向左移动5

本实例对机器人控制指令的输出结果进行模拟,将英文指令翻译为中文指令,在真实情况下,系统将调用不同的控制程序对机器人进行控制,包括对移动方向、方式和距离的控制等。

解释器模式的优缺点与适用环境

解释器模式为自定义语言的设计和实现提供了一种解决方案,用于定义一组文法规则并通过这组文法规则来解释语言中的句子。虽然解释器模式的使用频率不是特别高,但是它在正则表达式、XML文档解释等领域还是得到了广泛使用。

解释器模式的优点

解释器模式的主要优点如下:

  • (1)解释器模式易于改变和扩展文法。由于在解释器模式中使用类来表示语言的文法规则,因此可以通过继承等机制来改变或扩展文法。
  • (2)在解释器模式中,每一条文法规则都可以表示为一个类,因此可以方便地实现一个简单的语言。
  • (3)实现文法较为容易。在抽象语法树中每一个表达式结点类的实现方式都是相似的,这些类的代码编写都不会特别复杂,还可以通过一些工具自动生成结点类代码。
  • (4)增加新的解释表达式较为方便。如果用户需要增加新的解释表达式只需要对应增加一个新的终结符表达式或非终结符表达式类,原有表达式类代码无须修改,符合开闭原则。

解释器模式的缺点

解释器模式的主要缺点如下:

  • (1)解释器模式对于复杂文法难以维护。在解释器模式中,每一条规则至少需要定义一个类,因此如果一个语言包含太多的文法规则,类的个数将会急剧增加,从而导致系统难以管理和维护,此时可以考虑使用语法分析程序等方式来取代解释器模式。
  • (2)其执行效率较低。由于在解释器模式中使用了大量的循环和递归调用,因此在解释较为复杂的句子时其速度很慢,而且代码的调试过程也比较麻烦。

解释器模式的适用环境

在以下情况下可以考虑使用解释器模式:

  • (1)可以将一个需要解释执行的语言中的句子表示为一棵抽象语法树。
  • (2)一些重复出现的问题可以用一种简单的语言进行表达。
  • (3)一个语言的文法较为简单。对于复杂的文法,解释器模式中的文法类层次结构将变得很庞大而无法管理,此时最好使用语法分析程序生成器。
  • (4)执行效率不是关键问题。高效的解释器通常不是通过直接解释抽象语法树来实现的,而是需要将它们转换成其他形式,使用解释器模式的执行效率并不高。

本章小结

(1)解释器模式的目的是:给定一个语言,定义它的文法的一种表示,并定义一个解释器,这个解释器使用该表示来解释语言中的句子。解释器模式是一种类行为型模式。

(2)解释器模式包含抽象表达式、终结符表达式、非终结符表达式和环境类4个角色。其中,抽象表达式声明了抽象的解释操作,是所有终结符表达式和非终结符表达式的公共父类;终结符表达式是抽象表达式的子类,实现了与文法中的终结符相关联的解释操作;非终结符表达式也是抽象表达式的子类,实现了文法中非终结符的解释操作;环境类用于存储解释器之外的一些全局信息。

(3)解释器模式的主要优点包括易于改变和扩展文法,可以方便地实现一个简单的语言,实现文法较为容易,且增加新的解释表达式较为方便。其主要缺点是对于复杂文法难以维护,并且其执行效率较低。

(4)解释器模式适用的环境:可以将一个需要解释执行的语言中的句子表示为一棵抽象语法树;一些重复出现的问题可以用一种简单的语言进行表达;一个语言的文法较为简单;执行效率不是关键问题。

(5)用户可以使用文法规则来定义一个语言,还可以通过抽象语法树以图形方式直观地表示一个语言的构成,每一棵抽象语法树对应一个语言实例。

0%