主頁 > 知識庫 > PHP設(shè)計模式之工廠模式(Factory Pattern)的講解

PHP設(shè)計模式之工廠模式(Factory Pattern)的講解

熱門標(biāo)簽:電子圍欄 科大訊飛語音識別系統(tǒng) 團(tuán)購網(wǎng)站 Linux服務(wù)器 Mysql連接數(shù)設(shè)置 服務(wù)器配置 阿里云 銀行業(yè)務(wù)

面向?qū)ο缶幊讨?,工廠模式是我們最常用的實例化對象模式,工廠類就是一個專門用來創(chuàng)建其它對象的類,工廠類在多態(tài)性編程實踐中是非常重要的。它允許動態(tài)替換類,修改配置,會使應(yīng)用程序更加靈活。掌握工廠模式對Web開發(fā)是必不可少的,它會給你的系統(tǒng)帶來更大的可擴(kuò)展性和盡量少的修改量。

工廠模式通常用來返回類似接口的不同的類,工廠的一種常見用法就是創(chuàng)建多態(tài)的提供者。

通常工廠模式有一個關(guān)鍵的構(gòu)造,即一般被命名為factory的靜態(tài)方法。這個靜態(tài)方法可以接受任意數(shù)量的參數(shù),并且必須返回一個對象。

一個非常貼近生活的例子來告訴你什么是工廠模式

但是工廠模式真的是個累贅嗎?其實并不是!他能夠作為一種設(shè)計模式流傳至今,一定是有他的道理的!只不過我們看到的例子只能說明工廠模式是什么,并不能很好說明工廠模式的優(yōu)點,所以我們學(xué)會后并不知道為什么要使用工廠模式,以及什么時候應(yīng)該去使用工廠模式!

其實工廠模式在我們的現(xiàn)實生活中非常常見,下面我舉個生活中的例子,大家應(yīng)該就能明白工廠模式的用處在哪里了!

麥當(dāng)勞大家都吃過吧?我們?nèi)c餐的時候,我們可以點一個漢堡,一杯可樂,一個薯條。我們還可以點一杯可樂,一個薯條。點完之后點餐員會問我們一句還要別的嗎?你說不要了! 然后你的這一份餐就點完了,可以給錢了。咦,我們發(fā)現(xiàn)這是一個建造者模式(Builder Pattern)啊!

(ps:這確實是突然發(fā)現(xiàn)的,之前寫建造者模式那篇文章的時候并沒有想到這個例子)

基本的工廠類:

?php
 class Fruit {
 // 對象從工廠類返回
 }
 Class FruitFactory {
 public static function factory() {
  // 返回對象的一個新實例
  return new Fruit();
 }
 }
 // 調(diào)用工廠
 $instance = FruitFactory::factory();
?>

利用工廠類生產(chǎn)對象:

?php
class Example
{
  // The parameterized factory method
  public static function factory($type)
  {
    if (include_once 'Drivers/' . $type . '.php') {
      $classname = 'Driver_' . $type;
      return new $classname;
    } else {
      throw new Exception('Driver not found');
    }
  }
}
// Load a MySQL Driver
$mysql = Example::factory('MySQL');
// Load an SQLite Driver
$sqlite = Example::factory('SQLite');
?>

一個完整的工廠類:

下面的程序定義了一個通用的工廠類,它生產(chǎn)能夠保存你所有操作的空對象,你可以獲得一個實例,這些操作都在那個實例中了。

?php
  /**
   * Generic Factory class
   * This Factory will remember all operations you perform on it,
   * and apply them to the object it instantiates.
   */
  class FruitFactory {
    private $history, $class, $constructor_args;
    /**
     * Create a factory of given class. Accepts extra arguments to be passed to
     * class constructor.
     */
    function __construct( $class ) {
      $args = func_get_args();
      $this->class = $class;
      $this->constructor_args = array_slice( $args, 1 );
    }
    function __call( $method, $args ) {
      $this->history[] = array(
        'action'  => 'call',
        'method'  => $method,
        'args'  => $args
      );
    }
    function __set( $property, $value ) {
      $this->history[] = array(
        'action'  => 'set',
        'property'  => $property,
        'value'    => $value
      );
    }
    /**
     * Creates an instance and performs all operations that were done on this MagicFactory
     */
    function instance() {
      # use Reflection to create a new instance, using the $args 
      $reflection_object = new ReflectionClass( $this->class ); 
      $object = $reflection_object->newInstanceArgs( $this->constructor_args ); 
      # Alternative method that doesn't use ReflectionClass, but doesn't support variable
      # number of constructor parameters.
      //$object = new $this->class();
      # Repeat all remembered operations, apply to new object.
      foreach( $this->history as $item ) {
        if( $item['action'] == 'call' ) {
          call_user_func_array( array( $object, $item['method'] ), $item['args'] );
        }
        if( $item['action'] == 'set' ) {
          $object->{$item['property']} = $item['value'];
        }
      }
      # Done
      return $object;
    }
  }
  class Fruit {
    private $name, $color;
    public $price;
    function __construct( $name, $color ) {
      $this->name = $name;
      $this->color = $color;
    }
    function setName( $name ) {
      $this->name = $name;
    }
    function introduce() {
      print "Hello, this is an {$this->name} {$this->sirname}, its price is {$this->price} RMB.";
    }
  }
  # Setup a factory
  $fruit_factory = new FruitFactory('Fruit', 'Apple', 'Gonn');
  $fruit_factory->setName('Apple');
  $fruit_factory->price = 2;
  # Get an instance
  $apple = $fruit_factory->instance();
  $apple->introduce();
?>

工廠模式主要是為創(chuàng)建對象提供過渡接口,以便將創(chuàng)建對象的具體過程屏蔽隔離起來,達(dá)到提高靈活性的目的。

工廠模式可以分為三類:

  • 簡單工廠模式(Simple Factory)
  • 工廠方法模式(Factory Method)
  • 抽象工廠模式(Abstract Factory)

這三種模式從上到下逐步抽象,并且更具一般性。

簡單工廠模式又稱靜態(tài)工廠方法模式;從命名上就可以看出這個模式一定很簡單。它存在的目的很簡單:定義一個用于創(chuàng)建對象的接口。

工廠方法模式去掉了簡單工廠模式中工廠方法的靜態(tài)屬性,使得它可以被子類繼承。這樣在簡單工廠模式里集中在工廠方法上的壓力可以由工廠方法模式里不同的工廠子類來分擔(dān)。

工廠方法模式仿佛已經(jīng)很完美的對對象的創(chuàng)建進(jìn)行了包裝,使得客戶程序中僅僅處理抽象產(chǎn)品角色提供的接口。那我們是否一定要在代碼中遍布工廠呢?大可不必。也許在下面情況下你可以考慮使用工廠方法模式:

  • 當(dāng)客戶程序不需要知道要使用對象的創(chuàng)建過程。
  • 客戶程序使用的對象存在變動的可能,或者根本就不知道使用哪一個具體的對象。

總結(jié)

以上就是這篇文章的全部內(nèi)容了,希望本文的內(nèi)容對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,謝謝大家對腳本之家的支持。如果你想了解更多相關(guān)內(nèi)容請查看下面相關(guān)鏈接

您可能感興趣的文章:
  • PHP設(shè)計模式(九)外觀模式Facade實例詳解【結(jié)構(gòu)型】
  • PHP設(shè)計模式之外觀模式(Facade)入門與應(yīng)用詳解
  • thinkphp5.1框架中容器(Container)和門面(Facade)的實現(xiàn)方法分析
  • 詳解PHP中的外觀模式facade pattern
  • 學(xué)習(xí)php設(shè)計模式 php實現(xiàn)門面模式(Facade)
  • php設(shè)計模式 Facade(外觀模式)
  • PHP設(shè)計模式之PHP迭代器模式講解
  • 淺談PHP設(shè)計模式之門面模式Facade

標(biāo)簽:蚌埠 萍鄉(xiāng) 衡水 大理 衢州 廣元 江蘇 棗莊

巨人網(wǎng)絡(luò)通訊聲明:本文標(biāo)題《PHP設(shè)計模式之工廠模式(Factory Pattern)的講解》,本文關(guān)鍵詞  ;如發(fā)現(xiàn)本文內(nèi)容存在版權(quán)問題,煩請?zhí)峁┫嚓P(guān)信息告之我們,我們將及時溝通與處理。本站內(nèi)容系統(tǒng)采集于網(wǎng)絡(luò),涉及言論、版權(quán)與本站無關(guān)。
  • 相關(guān)文章
  • 收縮
    • 微信客服
    • 微信二維碼
    • 電話咨詢

    • 400-1100-266