热搜:NVER node 开发 php

PHP草根论之设计模式-策略模式

2024-08-20 10:10:01
PHP草根论之设计模式-策略模式

定义一系列的算法,把它们一个个封装起来, 并且使它们可相互替换。本模式使得算法的变化可独立于使用它的客户。

 

意图

 

策略模式

定义一系列的算法,把它们一个个封装起来, 并且使它们可相互替换。本模式使得算法可独立于使用它的客户而变化。

适用性

  • 许多相关的类仅仅是行为有异。“策略”提供了一种用多个行为中的一个行为来配置一个类的方法。
  • 需要使用一个算法的不同变体。例如,你可能会定义一些反映不同的空间/时间权衡的算法。当这些变体实现为一个算法的类层次时,可以使用策略模式。
  • 算法使用客户不应该知道的数据。可使用策略模式以避免暴露复杂的、与算法相关的数据结构。
  • 一个类定义了多种行为, 并且这些行为在这个类的操作中以多个条件语句的形式出现。将相关的条件分支移入它们各自的Strategy类中以代替这些条件语句
  •  

     

     

    关于模式本身的概念,请参考网上其它文章

    此处只讨论在PHP实际开发过程中的应用

     

     

    此模式将算法与数据进行分离.

    算法:狭义算法,对固定结构的输入数据产生一个固定结构的输出.  不同的算法,输入是相同的,输出是不同但类似的,或输出相同但实现过程不同.

    数据:一个持有数据的对象/类定义.

     

    应用此模式的场景:

        1.多种不同类的数据对象, 持有相同结构的数据.

        2.多种算法对象,同一类或实现了同一接口

        3.调用者先告诉数据对象将使用哪一个算法 对象.  然后数据对象在业务逻辑中将使用指定的算法 对象来对数据进行处理.

     

    使用局限:

        1.通常PHP不用于进行数学计算类的任务,这种任务通常由C/C++/Java来进行.

        2.多种数据对象,多种算法对象, 一般的PHP业务逻辑不会复杂到这种程度.

      

    设想一个实例: 程序中有多个业务处理类, 我们要对业务处理过程记录日志,  现在有多种日志方式可供选择:文本/数据库/网络应用.调用者可以自行选择使用哪种日志方式.

     

    实现过程:

        1.先定义一个日志的父类/接口,然后创建三种具体的日志实现类

            interface LogInterface {

               public function log($data);

            }

     

       class LogFile implements LogInterface{

          public function log($data){

    //将数据写入文本文件

    }

       }

     

       class LogDB implements LogInterface{

            public function log($data){

                   //将数据写入数据库

            }

        }

        class LogNet implements LogInterface{

                public function log($data){

                    //通过网络对数据进行日志记录

               }

        }

    2.定义一个Trait (为所有业务类增加一些功能)

    trait LogTrait{

        //记录具体的日志对象

        private $_log;

     

        //调用者用来设置具体的日志对象

        public function setLog(LogInterface $log){

           $this->_log=$log;

        }

     

    //业务类中使用的功能

    //此方法可以定义为私有,如果没有外部访问的话

    //此方法的实现也可变更,数据可以使用参数方式传递

    public function log(){

      $this->_log->log($this->data);

    }

    }

     

    3.所有的业务类(需要日志功能的)

     

    //此类可能已经继承了父类

    class ModelA {

    use LogTrait;

     

    //业务类中的某一段需要使用日志的代码段

    public function process(){

    //为日志准备数据

    $this->data=$data;

     

    //记录日志,此代码可在本类中多次出现,或被外部调用

    $this->log();

    }

    }