问题:
解释器模式就是用“迷你语言”来表现程序要解决的问题,以“迷你语言”写成“迷你程序”来表现具体的问题,例如正则表达式就是一个很好的例子。所以,通常当有一个语言需要解释执行,并且可以将该语言中的句子表示为一个抽象语法树时,就可以使用解释器模式了。
概念:
给定一个语言,定义它的文法的一种表示,并定义一个解释器,这个解释器使用该表示来解释语言中的句子。
实现:
1. 类图示例:
//表达文本类
class InterpreterContext
{
private $expressionstore = [];
public function replace(Expression $exp,$value)
{
$this->expressionstore[$exp->getKey()] = $value;
}
public function lookup(Expression $exp)
{
return $this->expressionstore[$exp->getKey()];
}
}
//表达式基类
abstract class Expression
{
private static $keycount = 0;
private $key;
abstract public function interpret(InterpreterContext $context);
public function getKey()
{
if(!isset($this->key)) {
self::$keycount ++;
$this->key = self::$keycount;
}
return $this->key;
}
}
//字符串表达式类
class LiteralExpression extends Expression
{
private $value;
public function __construct($value)
{
$this->value = $value;
}
public function interpret(InterpreterContext $context)
{
$context->replace($this,$this->value);
}
}
//变量表达式类
class VariableExpression extends Expression
{
private $name;
private $val;
public function __construct($name,$val = null)
{
$this->name = $name;
$this->val = $val;
}
public function interpret(InterpreterContext $context)
{
if(!is_null($this->val)) {
$context->replace($this,$this->val);
$this->val = null;
}
}
public function setValue($value)
{
$this->val = $value;
}
public function getKey()
{
return $this->name;
}
}
//操作符表达式基类
abstract class OperatorExpression extends Expression
{
protected $l_op;
protected $r_op;
public function __construct(Expression $l_op,Expression $r_op)
{
$this->l_op = $l_op;
$this->r_op = $r_op;
}
public function interpret(InterpreterContext $context)
{
$this->l_op->interpret($context);
$this->r_op->interpret($context);
$result_l = $context->lookup($this->l_op);
$result_r = $context->lookup($this->r_op);
$this->doInterpret($context,$result_l,$result_r);
}
abstract protected function doInterpret(InterpreterContext $context,$result_r);
}
//相等表达式类
class EqualsExpression extends OperatorExpression
{
protected function doInterpret(InterpreterContext $context,$result_r)
{
$context->replace($this,$result_l == $result_r);
}
}
//布尔或表达式类
class BooleanorExpression extends OperatorExpression
{
protected function doInterpret(InterpreterContext $context,$result_l || $result_r);
}
}
//布尔与表达式类
class BooleanAndExpression extends OperatorExpression
{
protected function doInterpret(InterpreterContext $context,$result_l && $result_r);
}
}
//应用:$input equals "4" or $input equals "four"
$context = new InterpreterContext();
$input = new VariableExpression(‘input‘);
$statement = new BooleanorExpression(
new EqualsExpression($input,new LiteralExpression(‘four‘)),new EqualsExpression($input,new LiteralExpression(‘4‘))
);
foreach([‘four‘,‘4‘,‘52‘] as $val) {
$input->setValue($val);
echo $val,‘:‘;
$statement->interpret($context);
if($context->lookup($statement)) {
echo ‘Yes<br>‘;
} else {
echo ‘No<br>‘;
}
}
效果:
创建了解释器的核心类后,解释器很容易进行扩展。但是解释器模式为文法中的每一条规则至少定义了一个类,当语言变得复杂,文法包含的规则很多时,需要创建的类的数量会很快增加,因此解释器模式最好应用于相对小的语言。
版权声明:本文内容由互联网用户自发贡献,该文观点与技术仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌侵权/违法违规的内容, 请发送邮件至 [email protected] 举报,一经查实,本站将立刻删除。