本文實(shí)例講述了PHP設(shè)計(jì)模式(一)工廠模式Factory。分享給大家供大家參考,具體如下:
在面向?qū)ο缶幊讨? 最通常的方法是一個(gè)new操作符產(chǎn)生一個(gè)對象實(shí)例,new操作符就是用來構(gòu)造對象實(shí)例的。但是在一些情況下, new操作符直接生成對象會帶來一些問題。舉例來說, 許多類型對象的創(chuàng)造需要一系列的步驟: 你可能需要計(jì)算或取得對象的初始設(shè)置; 選擇生成哪個(gè)子對象實(shí)例; 或在生成你需要的對象之前必須先生成一些輔助功能的對象。 在這些情況,新對象的建立就是一個(gè) “過程”,不僅是一個(gè)操作,像一部大機(jī)器中的一個(gè)齒輪傳動。
模式的問題:你如何能輕松方便地構(gòu)造對象實(shí)例,而不必關(guān)心構(gòu)造對象實(shí)例的細(xì)節(jié)和復(fù)雜過程呢?
解決方案:建立一個(gè)工廠來創(chuàng)建對象。
實(shí)現(xiàn):
一、引言
1)還沒有工廠時(shí)代:假如還沒有工業(yè)革命,如果一個(gè)客戶要一款寶馬車,一般的做法是客戶去創(chuàng)建一款寶馬車,然后拿來用。
2)簡單工廠模式:后來出現(xiàn)工業(yè)革命。用戶不用去創(chuàng)建寶馬車。因?yàn)榭蛻粲幸粋€(gè)工廠來幫他創(chuàng)建寶馬.想要什么車,這個(gè)工廠就可以建。比如想要320i系列車。工廠就創(chuàng)建這個(gè)系列的車。即工廠可以創(chuàng)建產(chǎn)品。
3)工廠方法模式時(shí)代:為了滿足客戶,寶馬車系列越來越多,如320i,523i,30li等系列一個(gè)工廠無法創(chuàng)建所有的寶馬系列。于是由單獨(dú)分出來多個(gè)具體的工廠。每個(gè)具體工廠創(chuàng)建一種系列。即具體工廠類只能創(chuàng)建一個(gè)具體產(chǎn)品。但是寶馬工廠還是個(gè)抽象。你需要指定某個(gè)具體的工廠才能生產(chǎn)車出來。
4)抽象工廠模式時(shí)代:隨著客戶的要求越來越高,寶馬車必須配置空調(diào)。而且這空調(diào)必須對應(yīng)給系列車才能使用。于是這個(gè)工廠開始生產(chǎn)寶馬車和需要的空調(diào)。
最終是客戶只要對寶馬的銷售員說:我要523i空調(diào)車,銷售員就直接給他523i空調(diào)車了。而不用自己去創(chuàng)建523i空調(diào)車寶馬車.
(我只是舉個(gè)例子,說到寶馬配置空調(diào)完全是為了舉例,甚至有點(diǎn)扯,哪有車和空調(diào)必須對應(yīng)才能使用?。?br />
這就是工廠模式。
二、分類
工廠模式主要是為創(chuàng)建對象提供過渡接口,以便將創(chuàng)建對象的具體過程屏蔽隔離起來,達(dá)到提高靈活性的目的。
工廠模式可以分為三類:
1)簡單工廠模式(Simple Factory)
2)工廠方法模式(Factory Method)
3)抽象工廠模式(Abstract Factory)
這三種模式從上到下逐步抽象,并且更具一般性。
GOF在《設(shè)計(jì)模式》一書中將工廠模式分為兩類:工廠方法模式(Factory Method)與抽象工廠模式(Abstract Factory)。將簡單工廠模式(Simple Factory)看為工廠方法模式的一種特例,兩者歸為一類。
三、區(qū)別
工廠方法模式:
一個(gè)抽象產(chǎn)品類,可以派生出多個(gè)具體產(chǎn)品類。
一個(gè)抽象工廠類,可以派生出多個(gè)具體工廠類。
每個(gè)具體工廠類只能創(chuàng)建一個(gè)具體產(chǎn)品類的實(shí)例。
抽象工廠模式:
多個(gè)抽象產(chǎn)品類,每個(gè)抽象產(chǎn)品類可以派生出多個(gè)具體產(chǎn)品類。
一個(gè)抽象工廠類,可以派生出多個(gè)具體工廠類。
每個(gè)具體工廠類可以創(chuàng)建多個(gè)具體產(chǎn)品類的實(shí)例。
區(qū)別:
工廠方法模式只有一個(gè)抽象產(chǎn)品類,而抽象工廠模式有多個(gè)。
工廠方法模式的具體工廠類只能創(chuàng)建一個(gè)具體產(chǎn)品類的實(shí)例,而抽象工廠模式可以創(chuàng)建多個(gè)。
兩者皆可。
四、簡單工廠模式
建立一個(gè)工廠(一個(gè)函數(shù)或一個(gè)類方法)來制造新的對象。
分布說明引子:從無到有??蛻糇约簞?chuàng)建寶馬車,然后拿來用。
![](http://img.jbzj.com/file_images/article/202005/20205284351443.jpg?20204284413)
?php
/**
* 車子系列
*
*/
Class BWM320{
function __construct($pa) {
}
}
Class BMW523{
function __construc($pb){
}
}
/**
*
* 客戶自己創(chuàng)建寶馬車
*/
class Customer {
function createBMW320(){
return new BWM320();
}
function createBMW523(){
return new BMW523();
}
}
客戶需要知道怎么去創(chuàng)建一款車,客戶和車就緊密耦合在一起了.為了降低耦合,就出現(xiàn)了工廠類,把創(chuàng)建寶馬的操作細(xì)節(jié)都放到了工廠里面去,客戶直接使用工廠的創(chuàng)建工廠方法,傳入想要的寶馬車型號就行了,而不必去知道創(chuàng)建的細(xì)節(jié).這就是工業(yè)革命了:簡單工廠模式
即我們建立一個(gè)工廠類方法來制造新的對象。如圖:
![](http://img.jbzj.com/file_images/article/202005/20205284426482.jpg?20204284453)
產(chǎn)品類:
?php
/**
* 車子系列
*
*/
abstract Class BWM{
function __construct($pa) {
}
}
Class BWM320 extends BWM{
function __construct($pa) {
}
}
Class BMW523 extends BWM{
function __construc($pb){
}
}
工廠類:
/**
*
* 工廠創(chuàng)建車
*/
class Factory {
static function createBMW($type){
switch ($type) {
case 320:
return new BWM320();
case 523:
return new BMW523();
//....
}
}
客戶類:
/**
*
* 客戶通過工廠獲取車
*/
class Customer {
private $BMW;
function getBMW($type){
$this¬-> BMW = Factory::createBMW($type);
}
}
簡單工廠模式又稱靜態(tài)工廠方法模式。重命名上就可以看出這個(gè)模式一定很簡單。它存在的目的很簡單:定義一個(gè)用于創(chuàng)建對象的接口。
先來看看它的組成:
1) 工廠類角色:這是本模式的核心,含有一定的商業(yè)邏輯和判斷邏輯。
2) 抽象產(chǎn)品角色:它一般是具體產(chǎn)品繼承的父類或者實(shí)現(xiàn)的接口。
3) 具體產(chǎn)品角色:工廠類所創(chuàng)建的對象就是此角色的實(shí)例。在java中由一個(gè)具體類實(shí)現(xiàn)。
下面我們從開閉原則(對擴(kuò)展開放;對修改封閉)上來分析下簡單工廠模式。當(dāng)客戶不再滿足現(xiàn)有的車型號的時(shí)候,想要一種速度快的新型車,只要這種車符合抽象產(chǎn)品制定的合同,那么只要通知工廠類知道就可以被客戶使用了。所以對產(chǎn)品部分來說,它是符合開閉原則的;但是工廠部分好像不太理想,因?yàn)槊吭黾右环N新型車,都要在工廠類中增加相應(yīng)的創(chuàng)建業(yè)務(wù)邏輯(createBMW($type)方法需要新增case),這顯然是違背開閉原則的。可想而知對于新產(chǎn)品的加入,工廠類是很被動的。對于這樣的工廠類,我們稱它為全能類 或者上帝類。
我們舉的例子是最簡單的情況,而在實(shí)際應(yīng)用中,很可能產(chǎn)品是一個(gè)多層次的樹狀結(jié)構(gòu)。由于簡單工廠模式中只有一個(gè)工廠類來對應(yīng)這些產(chǎn)品,所以這可能會把我們的上帝累壞了,也累壞了我們這些程序員:(
于是工廠方法模式作為救世主出現(xiàn)了。 工廠類定義成了接口,而每新增的車種類型,就增加該車種類型對應(yīng)工廠類的實(shí)現(xiàn),這樣工廠的設(shè)計(jì)就可以擴(kuò)展了,而不必去修改原來的代碼。
五、工廠方法模式
工廠方法模式去掉了簡單工廠模式中工廠方法的靜態(tài)屬性,使得它可以被子類繼承。這樣在簡單工廠模式里集中在工廠方法上的壓力可以由工廠方法模式里不同的工廠子類來分擔(dān)。
工廠方法模式組成:
1)抽象工廠角色: 這是工廠方法模式的核心,它與應(yīng)用程序無關(guān)。是具體工廠角色必須實(shí)現(xiàn)的接口或者必須繼承的父類。在java中它由抽象類或者接口來實(shí)現(xiàn)。
2)具體工廠角色:它含有和具體業(yè)務(wù)邏輯有關(guān)的代碼。由應(yīng)用程序調(diào)用以創(chuàng)建對應(yīng)的具體產(chǎn)品的對象。
3)抽象產(chǎn)品角色:它是具體產(chǎn)品繼承的父類或者是實(shí)現(xiàn)的接口。在java中一般有抽象類或者接口來實(shí)現(xiàn)。
4)具體產(chǎn)品角色:具體工廠角色所創(chuàng)建的對象就是此角色的實(shí)例。在java中由具體的類來實(shí)現(xiàn)。
工廠方法模式使用繼承自抽象工廠角色的多個(gè)子類來代替簡單工廠模式中的“上帝類”。正如上面所說,這樣便分擔(dān)了對象承受的壓力;而且這樣使得結(jié)構(gòu)變得靈活 起來——當(dāng)有新的產(chǎn)品產(chǎn)生時(shí),只要按照抽象產(chǎn)品角色、抽象工廠角色提供的合同來生成,那么就可以被客戶使用,而不必去修改任何已有 的代碼??梢钥闯龉S角色的結(jié)構(gòu)也是符合開閉原則的!
![](http://img.jbzj.com/file_images/article/202005/20205284624180.jpg?20204284647)
代碼如下:
產(chǎn)品類:
?php
/**
* 車子系列
*
*/
abstract Class BWM{
function __construct($pa) {
}
}
Class BWM320 extends BWM{
function __construct($pa) {
}
}
Class BMW523 extends BWM{
function __construc($pb){
}
}
創(chuàng)建工廠類:
/**
* 創(chuàng)建工廠的接口
*
*/
interface FactoryBMW {
function createBMW();
}
/**
*
* 創(chuàng)建BWM320車
*/
class FactoryBWM320 implements FactoryBMW {
function createBMW($type){
return new BWM320();
}
}
/**
*
* 創(chuàng)建BWM523車
*/
class FactoryBWM523 implements FactoryBMW {
function createBMW($type){
return new BMW523();
}
}
客戶類:
/**
*
* 客戶得到車
*/
class Customer {
private $BMW;
function getBMW($type){
switch ($type) {
case 320:
$BWM320 = new FactoryBWM320();
return $BWM320->createBMW();
case 523:
$BWM523 = new FactoryBWM523();
return $BWM320->createBMW();
//....
}
}
}
可以看出工廠方法的加入,使得對象的數(shù)量成倍增長。當(dāng)產(chǎn)品種類非常多時(shí),會出現(xiàn)大量的與之對應(yīng)的工廠對象,這不是我們所希望的。因?yàn)槿绻荒鼙苊膺@種情 況,可以考慮使用簡單工廠模式與工廠方法模式相結(jié)合的方式來減少工廠類:即對于產(chǎn)品樹上類似的種類(一般是樹的葉子中互為兄弟的)使用簡單工廠模式來實(shí) 現(xiàn)。
工廠方法小結(jié):
工廠方法模式仿佛已經(jīng)很完美的對對象的創(chuàng)建進(jìn)行了包裝,使得客戶程序中僅僅處理抽象產(chǎn)品角色提供的接口。那我們是否一定要在代碼中遍布工廠呢?大可不必。也許在下面情況下你可以考慮使用工廠方法模式:
1)當(dāng)客戶程序不需要知道要使用對象的創(chuàng)建過程。
2)客戶程序使用的對象存在變動的可能,或者根本就不知道使用哪一個(gè)具體的對象。
簡單工廠模式與工廠方法模式真正的避免了代碼的改動了?沒有。在簡單工廠模式中,新產(chǎn)品的加入要修改工廠角色中的判斷語句;而在工廠方法模式中,要么將判 斷邏輯留在抽象工廠角色中,要么在客戶程序中將具體工廠角色寫死(就象上面的例子一樣)。而且產(chǎn)品對象創(chuàng)建條件的改變必然會引起工廠角色的修改。
面對這種情況,我們可以使用反射機(jī)制:
class Customer {
private $BMW;
function getBMW($type){
$class = new ReflectionClass('FactoryBWM' .$type );//建立 'FactoryBWM'這個(gè)類的反射類
$instance = $class->newInstanceArgs();//相當(dāng)于實(shí)例化'FactoryBWM' .$type類
return $instance->createBMW();
//或者直接
/**
* $instance = new 'FactoryBWM' .$type();
* return $instance->createBMW();
*/
}
}
六、抽象工廠模式
隨著客戶的要求越來越高,寶馬車需要配置空調(diào)。于是這個(gè)工廠開始生產(chǎn)寶馬車和配置需要的空調(diào)。這時(shí)候工廠有二個(gè)系列的產(chǎn)品:寶馬車和空調(diào).寶馬車必須使用對應(yīng)的空調(diào)才能使用.這時(shí)候分別使用一個(gè)車工廠和一個(gè)空調(diào)工廠都不能滿足我們的需求,我們必須確認(rèn)車跟空調(diào)的對應(yīng)關(guān)系。因此把車工廠跟空調(diào)工廠聯(lián)系在一起。因此出現(xiàn)了抽象工廠模式。
可以說,抽象工廠模式和工廠方法模式的區(qū)別就在于需要創(chuàng)建對象的復(fù)雜程度上。而且抽象工廠模式是三個(gè)里面最為抽象、最具一般性的。
抽象工廠模式的用意為:給客戶端提供一個(gè)接口,可以創(chuàng)建多個(gè)產(chǎn)品族中的產(chǎn)品對象 ,而且使用抽象工廠模式還要滿足一下條件:
1)系統(tǒng)中有多個(gè)產(chǎn)品族,而系統(tǒng)一次只可能消費(fèi)其中一族產(chǎn)品。
2)同屬于同一個(gè)產(chǎn)品族的產(chǎn)品以其使用。
抽象工廠模式的各個(gè)角色(和工廠方法一樣):
1)抽象工廠角色: 這是工廠方法模式的核心,它與應(yīng)用程序無關(guān)。是具體工廠角色必須實(shí)現(xiàn)的接口或者必須繼承的父類。在java中它由抽象類或者接口來實(shí)現(xiàn)。
2)具體工廠角色:它含有和具體業(yè)務(wù)邏輯有關(guān)的代碼。由應(yīng)用程序調(diào)用以創(chuàng)建對應(yīng)的具體產(chǎn)品的對象。
3)抽象產(chǎn)品角色:它是具體產(chǎn)品繼承的父類或者是實(shí)現(xiàn)的接口。
4)具體產(chǎn)品角色:具體工廠角色所創(chuàng)建的對象就是此角色的實(shí)例。
其結(jié)構(gòu):
![](http://img.jbzj.com/file_images/article/202005/20205284839029.jpg?2020428493)
我們的例子:
![](http://img.jbzj.com/file_images/article/202005/20205284917159.jpg?20204284941)
代碼:
產(chǎn)品類:
?php
/**
* 車子系列以及型號
*
*/
abstract class BWM{
}
class BWM523 extends BWM {
}
class BWM320 extends BWM {
}
/**
* 空調(diào)
*
*/
abstract class aircondition{
}
class airconditionBWM320 extends aircondition {
}
class airconditionBWM52 extends aircondition {
}
創(chuàng)建工廠類:
/**
* 創(chuàng)建工廠的接口
*
*/
interface FactoryBMW {
function createBMW();
function createAirC();
}
/**
*
* 創(chuàng)建BWM320車
*/
class FactoryBWM320 implements FactoryBMW {
function createBMW(){
return new BWM320();
}
function createAirC(){ //空調(diào)
return new airconditionBWM320();
}
}
/**
*
* 創(chuàng)建BWM523車
*/
class FactoryBWM523 implements FactoryBMW {
function createBMW(){
return new BWM523();
}
function createAirC(){
return new airconditionBWM523();
}
}
客戶:
/**
*
* 客戶得到車
*/
class Customer {
private $BMW;
private $airC;
function getBMW($type){
$class = new ReflectionClass('FactoryBWM' .$type );//建立 Person這個(gè)類的反射類
$instance = $class->newInstanceArgs();//相當(dāng)于實(shí)例化Person 類
$this->BMW = $instance->createBMW();
$this->airC = $instance->createAirC();
}
}
更多關(guān)于PHP相關(guān)內(nèi)容感興趣的讀者可查看本站專題:《php面向?qū)ο蟪绦蛟O(shè)計(jì)入門教程》、《PHP數(shù)組(Array)操作技巧大全》、《PHP基本語法入門教程》、《PHP運(yùn)算與運(yùn)算符用法總結(jié)》、《php字符串(string)用法總結(jié)》、《php+mysql數(shù)據(jù)庫操作入門教程》及《php常見數(shù)據(jù)庫操作技巧匯總》
希望本文所述對大家PHP程序設(shè)計(jì)有所幫助。
您可能感興趣的文章:- PHP設(shè)計(jì)模式之原型模式示例詳解
- PHP設(shè)計(jì)模式之命令模式示例詳解
- PHP設(shè)計(jì)模式(四)原型模式Prototype實(shí)例詳解【創(chuàng)建型】
- PHP設(shè)計(jì)模式(三)建造者模式Builder實(shí)例詳解【創(chuàng)建型】
- PHP設(shè)計(jì)模式概論【概念、分類、原則等】
- PHP設(shè)計(jì)模式之 策略模式Strategy詳解【對象行為型】
- PHP設(shè)計(jì)模式入門之狀態(tài)模式原理與實(shí)現(xiàn)方法分析
- PHP設(shè)計(jì)模式入門之迭代器模式原理與實(shí)現(xiàn)方法分析
- PHP設(shè)計(jì)模式之迭代器模式Iterator實(shí)例分析【對象行為型】
- php設(shè)計(jì)模式之適配器模式實(shí)例分析【星際爭霸游戲案例】
- php設(shè)計(jì)模式之迭代器模式實(shí)例分析【星際爭霸游戲案例】
- 詳解PHP八大設(shè)計(jì)模式