zoukankan      html  css  js  c++  java
  • PHP设计模式—解释器模式

    定义:

    解释器模式(interpreter):给定一个语言,定义它的文法的一种表示,并定义一个解释器,这个解释器使用该表示来解释语言中的句子。解释器模式需要解决的是,如果一种特定类型的问题发生的频率足够高,那么可能就值得将该问题的各个实例表述为一个简单语言中的句子。这样就可以构建一个解释器,该解释器通过解释这些句子来解决该问题。

    结构:

    • AbstractExpression(抽象表达式):表明一个抽象的解释操作,这个接口为抽象语法树中所有的节点所共享。
    • TerminalExpression(终结符表达式):实现与文法中的终结符相关联的解释操作,实现抽象表达式中所要求的接口,文法中每一个终结符都有一个具体终结表达式与之相对应。
    • NonterminalExpression(非终结符表达式):为文法中的非终结符实现解释操作。
    • Context:包含解释器之外的一些全局信息。
    • Client:客户端代码,构建表示该文法定义的语言中一个特定的句子的抽象语法树,调用解释操作。

    代码实例:

    用解释器模式设计一个“韶粵通”公交车卡的读卡器程序。
    说明:假如“韶粵通”公交车读卡器可以判断乘客的身份,如果是“韶关”或者“广州”的“老人” “妇女”“儿童”就可以免费乘车,其他人员乘车一次扣 2 元。

    // Expression.php
    /**
     * 抽象表达类
     * Class Expression
     */
    abstract class Expression
    {
        abstract public function interpret($info);
    }
    
    // TerminalExpression.php
    /**
     * 终结符表达式类
     * Class TerminalExpression
     */
    class TerminalExpression extends Expression
    {
        private $value = [];
    
        public function __construct(array $value)
        {
            $this->value = $value;
        }
    
        public function interpret($info)
        {
            // TODO: Implement interpret() method.
            if (in_array($info, $this->value)) {
                return true;
            } else {
                return false;
            }
        }
    }
    
    // AndExpression.php
    /**
     * 非终结符表达式类
     * Class AndExpression
     */
    class AndExpression extends Expression
    {
        private $city;
        private $person;
    
        public function __construct(Expression $city, Expression $person)
        {
            $this->city = $city;
            $this->person = $person;
        }
    
        public function interpret($info)
        {
            // TODO: Implement interpret() method.
            $data = explode('的', $info);
            return $this->city->interpret($data[0]) && $this->person->interpret($data[1]);
        }
    }
    
    // Context.php
    /**
     * Class Context
     */
    class Context
    {
        private $city = ["韶关", "广州"];
        private $person = ["老人", "妇女", "儿童"];
        private $cityPerson;
    
        public function __construct()
        {
            $cityExpression = new TerminalExpression($this->city);
            $personExpression = new TerminalExpression($this->person);
            $this->cityPerson = new AndExpression($cityExpression, $personExpression);
        }
    
        public function freeRide($info)
        {
            $isTrue = $this->cityPerson->interpret($info);
            if ($isTrue) {
                echo "您是" . $info . ",您本次乘车免费!
    ";
            } else {
                echo $info . ",您不是免费人员,本次乘车扣费2元!
    ";
            }
        }
    }

    客户端调用:

    $context = new Context();
    $context->freeRide("韶关的老人");
    $context->freeRide("韶关的年轻人");
    $context->freeRide("广州的妇女");
    $context->freeRide("广州的儿童");
    $context->freeRide("山东的儿童");

    结果:

    您是韶关的老人,您本次乘车免费!
    韶关的年轻人,您不是免费人员,本次乘车扣费2元!
    您是广州的妇女,您本次乘车免费!
    您是广州的儿童,您本次乘车免费!
    山东的儿童,您不是免费人员,本次乘车扣费2元!

    总结:

    • 通常当有一个语言需要解释执行,并且你可将该语言中的句子表示为一个抽象语法树时,可使用解释器模式。
    • 使用解释器模式,意味着可以很容易的改变和扩展文法,因为该模式使用类来表示文法规则,你可以使用继承来改变或扩展该文法。也比较容易实现文法,因为定义抽象语法树中各个节点的类的实现大体类似,这些类都易于直接编写。
    • 解释器模式也有不足的地方,解释器模式为文法中的每一条规则至少定义了一个类,因此包含许多规则的文法可能难以管理和维护。
  • 相关阅读:
    设计模式之适配器模式温故知新(九)
    设计模式之策略模式总结(八)
    设计模式之观察者模式, 个人感觉相当的重要(七)
    设计模式之抽象工厂模式读后(六)
    设计模式之工厂模式详细读后感TT!(五)
    设计模式之简单工厂模式, 加速(四)
    设计模式之代理模式笔记(三)
    设计模式之单例模式读后思考(二)
    为什么要用设计模式?先看看6大原则(一)
    Codeforces_835
  • 原文地址:https://www.cnblogs.com/woods1815/p/13550955.html
Copyright © 2011-2022 走看看