首頁 >後端開發 >php教程 >詳解PHP物件導向五大原則之單一職責原則

詳解PHP物件導向五大原則之單一職責原則

零到壹度
零到壹度原創
2018-04-11 10:52:491912瀏覽

本文實例講述了PHP物件導向五大原則之單一職責原則(SRP)。分享給大家供大家參考,具體如下:

單一職責原則(Single Pesponsibility Principle, SRP)

單一職責有兩個意義: 一個是避免相同的職責分散到不同的類別中, 別一個是避免一個類別承擔太多職責

為什麼要遵守SRP呢?

(1)可以減少類別之間的耦合

如果減少類別之間的耦合,當需求變化時,只修改一個類,從而也就隔離了變化;如果一個類別有多個不同職責,它們耦合在一起,當一個職責發生變化時,可能會影響到其他職責。

(2)提高類別的複用性

修改電腦比修理電視機簡單多了。主要原因就在於電視機各個部件之間的耦合性太高,而電腦則不同,電腦的內存、硬碟、音效卡、網卡、鍵盤燈等部件都可以輕鬆地單獨拆卸和組裝。某個部件壞了,換上新的即可。上面的例子就體現了單一職責的優勢。由於使用了單一職責,使得‘組件'可以方便地‘拆卸’和‘組裝’。

不遵守SRP會影響對類別的複用性。當只需要用該類別的某一個職責時,由於它和其他的職責耦合在一起,也就很難分離出。

遵守SRP在實際程式碼開發上有沒有什麼應用?有的。以資料持久層為例,所謂的資料持久層主要指的是資料庫操作,當然,還包括快取管理等。這時就需要資料持久層支援多種資料庫。應該怎麼做?定義多個資料庫操作類別?想法已經很接近了,再進一步,就是使用工廠模式。

工廠模式(Faction)允許你在程式碼執行時實例化物件。它之所以被稱為工廠模式是因為它負責『生產對象』。以資料庫為例,工廠需要的就是根據不同的參數,產生不同的實例化物件。最簡單的工廠就是根據傳入的型別名稱實例化對象,如傳入MySQL,就呼叫MySQL類別並實例化,如果是SQLite,則呼叫SQLite的類別並實例化,甚至還可以處理TXT、Execl等'類資料庫'。

工廠類也就是這樣的一個類,它只負責生產對象,而不負責對象的具體內容。

以下是範例

定義一個適配器的介面

  1. interface Db_Adpater
    {
      /**
       * 数据库连接
       * @param $config 数据库配置
       * @return mixed resource
       */
      public function connect($config);
      /**
       * 执行数据库查询
       * @param $query 数据库查询的SQL字符串
       * @param $handle 连接对象
       * @return mixed
       */
      public function query($query,$handle);
    }

定義一個實作了DB_Adpater介面的MySQL資料庫操作類

  1. class Db_Adapter_Mysql implements Db_Adpater
    {
      private $_dbLink;  //数据库连接字符串标识
      /**
       * 数据库连接函数
       * @param $config 数据库配置
       * @return resource
       * @throws Db_Exception
       */
      public function connect($config)
      {
        if($this->_dbLink = @mysql_connect($config->host . (empty($config->port) ? '' : ':' . $config->prot) ,$config->user, $config->password, true))
        {
          if(@mysql_select_db($config->database, $this->_dbLink))
          {
            if($config->charset)
            {
              mysql_query("SET NAME '{$config->charset}'", $this->_dbLink);
            }
            return $this->_dbLink;
          }
        }
        throw new Db_Exception(@mysql_error($this->_dbLink));
      }
      /**
       * 执行数据库查询
       * @param $query 数据库查询SQL字符串
       * @param $handle 连接对象
       * @return resource
       */
      public function query($query,$handle)
      {
        if($resource = @mysql_query($query,$handle))
          return $resource;
      }
    }

定義一個實作了DB_Adpater介面的SQLite資料庫操作類別

  1. class Db_Adapter_sqlite implements Db_Adpater
    {
      private $_dbLink;  //数据库连接字符串标识
      public function connect($config)
      {
        if($this->_dbLink = sqlite_open($config->file, 0666, $error))
        {
          return $this->_dbLink;
        }
        throw new Db_Exception($error);
      }
      public function query($query, $handle)
      {
        if($resource = @sqlite_query($query,$handle))
        {
          return $resource;
        }
      }
    }

#現在如果需要一個資料庫操作的方法怎麼做,只需定義一個工廠類,根據傳入不同的生成需要的類即可

  1. #
    class sqlFactory
    {
      public static function factory($type)
      {
        if(include_once 'Drivers/' . $type . '.php')
        {
          $classname = 'Db_Adapter_'.$type;
          return new $classname;
        }
        else
          throw new Exception('Driver not found');
      }
    }

調用時,就可以這麼寫

  1. $db = sqlFactory::factory('MySQL');
    $db = sqlFactory::factory('SQLite');

我们把创建数据库连接这块程序单独拿出来,程序中的CURD就不用关心什么数据库了,只要按照规范使用对应的方法即可。

工厂方法让具体的对象解脱出来,使其不再依赖具体的类,而是抽象。

设计模式里面的命令模式也是SRP的体现,命令模式分离“命令的请求者”和“命令的实现者”方面的职责。举一个很好理解的例子,就是你去餐馆订餐吃饭,餐馆存在顾客、服务员、厨师三个角色。作为顾客,你要列出菜单,传给服务员,由服务员通知厨师去实现。作为服务员,只需要调用准备饭菜这个方法(对厨师喊“该炒菜了”),厨师听到要炒菜的请求,就立即去做饭。在这里,命令的请求和实现就完成了解耦。

模拟这个过程,首先定义厨师角色,厨师进行实际做饭、烧汤的工作。

以下是示例

  1. /**
     * 厨师类,命令接受者与执行者
     * Class cook
     */
    class cook
    {
      public function meal()
      {
        echo '番茄炒鸡蛋',PHP_EOL;
      }
      public function drink()
      {
        echo '紫菜蛋花汤',PHP_EOL;
      }
      public function ok()
      {
        echo '完毕',PHP_EOL;
      }
    }
    //然后是命令接口
    interface Command
    {
      public function execute();
    }

轮到服务员出场,服务员是命令的传送者,通常你到饭馆吃饭都是叫服务员吧,不能直接叫厨师,一般都是叫“服务员,给我来盘番茄炒西红柿”。所以,服务员是顾客和厨师之间的命令沟通都。

  1. class MealCommand implements Command
    {
      private $cook;
      //绑定命令接受者
      public function __construct(cook $cook)
      {
        $this->cook = $cook;
      }
      public function execute()
      {
        $this->cook->meal();//把消息传给厨师,让厨师做饭,下同
      }
    }
    class DrinkCommand implements Command
    {
      private $cook;
      //绑定命令接受者
      public function __construct(cook $cook)
      {
        $this->cook = $cook;
      }
      public function execute()
      {
        $this->cook->drink();
      }
    }

现在顾客可以按照菜单叫服务员了

  1. class cookControl
    {
      private $mealcommand;
      private $drinkcommand;
      //将命令发送者绑定以命令接收器上面来
      public function addCommand(Command $mealcommand, Command $drinkcommand)
      {
        $this->mealcommand = $mealcommand;
        $this->drinkcommand = $drinkcommand;
      }
      public function callmeal()
      {
        $this->mealcommand->execute();
      }
      public function calldrink()
      {
        $this->drinkcommand->execute();
      }
    }

好了,现在完成整个过程

  1. $control = new cookControl;
    $cook = new cook;
    $mealcommand = new MealCommand($cook);
    $drinkcommand = new DrinkCommand($cook);
    $control->addCommand($mealcommand,$drinkcommand);
    $control->callmeal();
    $control->calldrink();

从上面的例子可以看出,原来设计模式并非纯理论的东西,而是来源于实际生活,就连普通的餐馆老板都懂设计模式这门看似高深的学问。其实,在经济和管理活动中对流程的优化就是对各种设计模式的摸索和实践。所以,设计模式并非计算机编程中的专利。事实上,设计模式的起源并不是计算机,而是源于建筑学。

在设计模式方面,不仅以上这两种体现了SRP,还有别的(比如代理模式)也体现了SRP。SRP不只是对类设计有意义,对以模块、子系统为单位的系统架构设计同样有意义。

模块、子系统也应该仅有一个引起它变化的原因,如MVC所倡导的各个层之间的相互分离就是SRP在系统总体设计中的应用。

SRP是最简单的原则之一,也是最难做好的原则之一。我们会很自然地将职责连接在一起。找到并且分离这些职责是软件设计需要达到的目的

一些简单的应用遵循的做法如下:

根据业务流程,把业务对象提炼出来。如果业务的流程的链路太复杂,就把这个业务对象分离为多个单一业务对象。当业务链标准化后,对业务对象的内部情况做进一步处理,把第一次标准化视为最高层抽象,第二次视为次高层抽象,以此类推,直到“恰如其分”的设计层次

职责的分类需要注意。有业务职责,还要有脱离业务的抽象职责,从认识业务到抽象算法是一个层层递进的过程。就好比命令模式中的顾客,服务员和厨师的职责,作为老板(即设计师)的你需要规划好各自的职责范围,即要防止越俎代庖,也要防止互相推诿。

以上是詳解PHP物件導向五大原則之單一職責原則的詳細內容。更多資訊請關注PHP中文網其他相關文章!

陳述:
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn