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