一、概述
在软件开发特别是DSL开发中常常需要使用一些相对较复杂的业务语言,如果业务语言使用频率足够高,且使用普通的编程模式来实现会导致非常复杂的变化,那么就可以考虑使用解释器模式构建一个解释器对复杂的业务语言进行翻译。这种做法虽然效率相对较低,但可以允许用户使用自定义的业务语言来处理逻辑,因此在效率不是关键问题的场合还是较为有用的。
二、解释器模式
解释器模式描述了如何为简单的语言定义一个文法,如何在该语言中表示一个句子,以及如何解释这些句子。其结构图如下:
AbstractExpression声明一个为抽象语法树中所有节点所共享的解释接口。
TerminalExpression表示终结符表达式,实现文法中终结符相关联的所有解释操作。
NonterminalExpression表示非终结符表达式,实现文法中所有非终结符的解释操作。
Context包含解释器外的一些全局信息。
Client构建语法树并调用解释器。
三、示例
我们我们就以中英文翻译系统为例来介绍解释器模式。
中英文对照字典如下。
public static class ChineseEnglishDict
{
private static Dictionary<string, string> _dictory = new Dictionary<string, string>(); static ChineseEnglishDict()
{
_dictory.Add("this", "这");
_dictory.Add("is", "是");
_dictory.Add("an", "一个");
_dictory.Add("apple", "苹果");
} public static string GetEnglish(string value)
{
return _dictory[value];
}
}
首先定义AbstractExpression接口。
public interface IExpression
{
void Interpret(StringBuilder sb);
}
接着定义具体的Expression,这里包括对英文单词的翻译和对英文句号的翻译。
public class WordExpression : IExpression
{
private string _value; public WordExpression(string value)
{
_value = value;
} public void Interpret(StringBuilder sb)
{
sb.Append(ChineseEnglishDict.GetEnglish(_value.ToLower()));
}
} public class SymbolExpression : IExpression
{
private string _value; public SymbolExpression(string value)
{
_value = value;
} public void Interpret(StringBuilder sb)
{
switch (_value)
{
case "." :
sb.Append("。");
break;
}
}
}
最后将各部分解释器组合起来进行包装,方便用户调用。
public static class Translator
{
public static string Translate(string sentense)
{
StringBuilder sb = new StringBuilder();
List<IExpression> expressions = new List<IExpression>();
string [] elements = sentense.Split(new char [] {'.'}, StringSplitOptions.RemoveEmptyEntries);
foreach (string element in elements)
{
string[] words = element.Split(new char [] {' '}, StringSplitOptions.RemoveEmptyEntries);
foreach (string word in words)
{
expressions.Add(new WordExpression(word));
}
expressions.Add(new SymbolExpression("."));
}
foreach (IExpression expression in expressions)
{
expression.Interpret(sb);
}
return sb.ToString();
}
}
让我们来看一下如何调用解释器。
static void Main(string[] args)
{
string englist = "This is an apple.";
string chinese = Translator.Translate(englist);
Console.WriteLine(chinese);
}