本文實(shí)例講述了PHP設(shè)計模式之工廠模式定義與用法。分享給大家供大家參考,具體如下:
工廠模式(Factory Design Pattern)作為一種創(chuàng)建型設(shè)計模式, 遵循了開放-封閉原則, 對修改封閉, 對擴(kuò)展開放. 工廠方法(Factory Method)模式就是要創(chuàng)建"某種東西". 對于工廠方法模式, 要創(chuàng)建的"東西"是一個產(chǎn)品,這個產(chǎn)品與創(chuàng)建它的類之間不存在綁定.實(shí)際上,為了保持這種松耦合,客戶會通過一個工廠發(fā)出請求. 再由工廠創(chuàng)建所請求的產(chǎn)品.也可以換種方式考慮, 利用工廠方法模式, 請求者只發(fā)出請求, 而不具體創(chuàng)建產(chǎn)品.
工廠的工作
先建立一個工廠的接口
Factory.php
?php
abstract class Factory
{
//抽象的創(chuàng)建對象的方法
protected abstract function createProduct();
//該方法調(diào)用createProduct方法返回一個產(chǎn)品對象.
public function start()
{
return $this->createProduct();
}
}
start
方法返回一個產(chǎn)品,該方法調(diào)用createProduct
方法完成產(chǎn)生產(chǎn)品的操作.所以createProduct的具體實(shí)現(xiàn)要構(gòu)建并返回一個按Product接口實(shí)現(xiàn)的產(chǎn)品對象.
比如產(chǎn)品都有一個共同的方法getProperties()
, 以下是對應(yīng)Product接口
Product.php
?php
//產(chǎn)品接口
interface Product
{
public function getProperties();
}
接著, 我們要建立兩個工廠,文本工廠TextFactory和圖像工廠phptoFactory
TextFactory.php
?php
include_once('Factory.php');
include_once('TextProduct.php');
class TextFactory extends Factory
{
protected function createProduct()
{
$product = new TextProduct();
return $product->getProperties();
}
}
PhotoFactory.php
?php
include_once('Factory.php');
include_once('PhotoProduct.php');
class PhotoFactory extends Factory
{
protected function createProduct()
{
$product = new PhotoProduct();
return $product->getProperties();
}
}
可以看到,在工廠方法的實(shí)現(xiàn)中, getProperties方法引入了多態(tài)(polymorphism), 將用這個方法返回"文本"或"圖像". 同一個getProperties()
有多個(poly)不同的形態(tài)(morphs), 這就是多態(tài).在這種情況下, 其中一種形式返回文本, 而另一種返回圖像.
可以在properties這個實(shí)現(xiàn)中放入你想要的任何東西,工廠方法設(shè)計將會創(chuàng)建這個對象, 并把他返回給Client使用.
下面的是兩個產(chǎn)品的實(shí)現(xiàn)
TextProduct.php
?php
include_once('Product.php');
class TextProduct implements Product
{
public function getProperties()
{
return "這里是文本產(chǎn)品";
}
}
PhotoProduct.php
?php
include_once('Product.php');
class PhotoProduct implements Product
{
//這是產(chǎn)品具有的方法
public function getProperties()
{
return "這里是圖像產(chǎn)品";
}
}
這兩個產(chǎn)品實(shí)現(xiàn)了Product接口中的抽象方法getProperties()
,
客戶(Client)
我們并不希望客戶直接做出產(chǎn)品請求.實(shí)際上, 我們希望客戶通過Factory工廠接口做出請求.這樣一來,如果以后我們增加了產(chǎn)品或者工廠, 客戶可以做同樣的請求來得到更多類型的產(chǎn)品 , 而不會破壞這個應(yīng)用:
Client.php
?php
include_once('PhotoFactory.php');
include_once('TextFactory.php');
class Client
{
public function __construct()
{
$this->somePhotoObject = new PhotoFactory();
echo $this->somePhotoObject->start() . 'br />';
$this->someTextObject = new TextFactory();
echo $this->someTextObject->start() . 'br />';
}
}
$worker = new Client();
運(yùn)行Client.php, 得到下面的結(jié)果
這里是圖像產(chǎn)品
這里是文本產(chǎn)品
注意: Client對象并沒有向產(chǎn)品直接做出請求, 而是通過工廠來請求. 重要的是, 客戶并不實(shí)現(xiàn)產(chǎn)品特性, 而留給產(chǎn)品實(shí)現(xiàn)來體現(xiàn).
調(diào)整產(chǎn)品
設(shè)計模式的真正價值并不是提高操作的速度, 而是加快開發(fā)的速度.
如果現(xiàn)在需求變化了, 需要對圖像產(chǎn)品做出修改, 只需要修改相應(yīng)的產(chǎn)品PhotoProduct的getProperties方法即可
對象的改變看起來很簡單 不過Product的getProperties()
方法仍保持相同的接口,請求工廠返回一個屬性對象
增加新產(chǎn)品和參數(shù)化請求
問題來了,如果要增加更多的圖像和文本說明, 有沒有必要每次增加一個新的區(qū)域就增加一個新的具體的工廠類?這意味著要為每個新區(qū)域增加一個新工廠和產(chǎn)品.于是,我們引進(jìn)了參數(shù)化工廠設(shè)計模式
參數(shù)化工廠設(shè)計模式和一般的工廠設(shè)計模式的主要區(qū)別之一是客戶包含工廠和產(chǎn)品的引用. 在參數(shù)化請求中, Client類必須指定產(chǎn)品, 而不是產(chǎn)品工廠. createProduct()
操作中的參數(shù)是由客戶傳入一個產(chǎn)品; 所以客戶必須指出它想要的具體產(chǎn)品. 不過, 這個請求仍然是通過工廠接口Factory發(fā)出的. 所以, 盡管客戶包含一個產(chǎn)品引用, 但通過Factory, 客戶仍然與產(chǎn)品分離.
一個工廠多個產(chǎn)品(參數(shù)化工廠方法)
對于大多數(shù)請求, 參數(shù)化工廠方法更為簡單, 因?yàn)榭蛻糁恍枰幚硪粋€具體工廠.工廠方法操作有一個參數(shù),指示需要創(chuàng)建的產(chǎn)品.而在原來的設(shè)計中, 每個產(chǎn)品都有自己的工廠, 不需要另個傳遞參數(shù); 產(chǎn)品實(shí)現(xiàn)依賴于各個產(chǎn)品特定的工廠.
新工廠接口
Factory.php
?php
abstract class Factory
{
//抽象的創(chuàng)建對象的方法
protected abstract function createProduct(Product $product);
//該方法由factoryMethod方法返回一個產(chǎn)品對象.
public function start($product)
{
return $this->createProduct($product);
}
}
在這個新的Factory接口中可以看到, create()
和start()
都需要一個參數(shù),指定一個Product對象, 而不是Product接口的一個特定實(shí)現(xiàn), 所以可以接受任何Product的具體實(shí)例.
工廠具體實(shí)現(xiàn)
具體的創(chuàng)建者類CommonFactory實(shí)現(xiàn)了createProduct()
,如下
CommonFactory.php
?php
include_once('Factory.php');
include_once('Product.php');
class CommonFactory extends Factory
{
protected function createProduct(Product $product)
{
return $product->getProperties();
}
}
這個類調(diào)用Product的方法getProperties將產(chǎn)品返回給客戶.
新產(chǎn)品
具體產(chǎn)品的變化并不會改變原來的Product接口,還是原來的代碼
?php
//產(chǎn)品接口
interface Product
{
public function getProperties();
}
例如, 現(xiàn)在有一個鋼筆產(chǎn)品PenProduct
PenProduct.php
?php
include_once('Product.php');
class PenProduct implements Product
{
public function getProperties()
{
return "鋼筆產(chǎn)品";
}
}
客戶Clent(有參數(shù))
?php
include_once('CommonFactory.php');
include_once('PenProduct.php');
class Client
{
public function __construct()
{
$commonFactory = new CommonFactory();
echo $commonFactory->start(new PenProduct());
}
}
$worker = new Client();
運(yùn)行后輸出
鋼筆產(chǎn)品
以后如果開發(fā)出了新的產(chǎn)品, 只需要創(chuàng)建對應(yīng)的產(chǎn)品類, 然后客戶指定想要的新產(chǎn)品 , 即可返回客戶需要的產(chǎn)品.
總結(jié):
產(chǎn)品改變: 接口不變
使用設(shè)計模式的一大好處就是可以很容易地對類做出改變, 而不會破壞更大的程序. 之所以能夠容易地做出改變, 秘訣在于保持接口不變, 而只改變內(nèi)容.
更多關(guān)于PHP相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《php面向?qū)ο蟪绦蛟O(shè)計入門教程》、《PHP基本語法入門教程》、《PHP數(shù)組(Array)操作技巧大全》、《php字符串(string)用法總結(jié)》、《php+mysql數(shù)據(jù)庫操作入門教程》及《php常見數(shù)據(jù)庫操作技巧匯總》
希望本文所述對大家PHP程序設(shè)計有所幫助。
您可能感興趣的文章:- php單態(tài)設(shè)計模式(單例模式)實(shí)例
- php設(shè)計模式 Singleton(單例模式)
- 基于php設(shè)計模式中單例模式的應(yīng)用分析
- 學(xué)習(xí)php設(shè)計模式 php實(shí)現(xiàn)單例模式(singleton)
- PHP設(shè)計模式之單例模式原理與實(shí)現(xiàn)方法分析
- PHP設(shè)計模式之抽象工廠模式實(shí)例分析
- PHP設(shè)計模式之簡單工廠和工廠模式實(shí)例分析
- PHP設(shè)計模式之模板方法模式實(shí)例淺析
- PHP設(shè)計模式之單例模式定義與用法分析