工廠模式
一個(gè)類(lèi)可能在很多地方都需要實(shí)例化。比如數(shù)據(jù)庫(kù)類(lèi)掷豺,每次使用都需要實(shí)例化。如果類(lèi)名改了薄声,則相對(duì)調(diào)用位置的代碼也需要修改当船。
因?yàn)轭?lèi)在實(shí)例化對(duì)象的時(shí)候,是用 new類(lèi)名 實(shí)現(xiàn)的奸柬。如果類(lèi)名修改生年,則所有的new 操作的確都需要重新修改代碼。
如果有一種方式廓奕,集中所有的實(shí)例化抱婉。這樣代碼就只需要修改一次档叔。那么代碼的修改就變得簡(jiǎn)單了。這就是工廠模式設(shè)計(jì)的最初的設(shè)計(jì)想法蒸绩。
核心思想就是:
1衙四、實(shí)例化對(duì)象不用new ,而用工廠方法代替
2患亿、將選擇實(shí)現(xiàn)類(lèi)和創(chuàng)建對(duì)象統(tǒng)一管理和控制传蹈。從而達(dá)到調(diào)用者和實(shí)現(xiàn)類(lèi)解耦的目的。
工廠模式分類(lèi)
簡(jiǎn)單工廠
用來(lái)生產(chǎn)同一結(jié)構(gòu)中的任意產(chǎn)品(對(duì)新增加產(chǎn)品步藕,需要修改原代碼)工廠方法
用來(lái)生產(chǎn)同一結(jié)構(gòu)產(chǎn)品(支持增加任意產(chǎn)品)抽象工廠
圍繞一個(gè)超級(jí)工廠創(chuàng)建其他工廠惦界,該超級(jí)工廠又稱(chēng)之為工廠的工廠。
工廠模式設(shè)計(jì)
定義 :工廠模式咙冗,顧名思義沾歪,就是像工廠流水線一樣生成對(duì)象。由一個(gè)地方產(chǎn)生對(duì)象雾消,其他地方就不需要額外的實(shí)例化灾搏。從而達(dá)到后期代碼統(tǒng)一維護(hù)的目的。而且工廠模式可以方便因此真實(shí)結(jié)構(gòu)立润,因此更加安全狂窑。從而實(shí)現(xiàn)創(chuàng)建者和調(diào)用者分離的目的。
特點(diǎn):將調(diào)用者和創(chuàng)建者分離桑腮,調(diào)用者直接向工廠類(lèi)請(qǐng)求獲取調(diào)用對(duì)象泉哈,減少代碼耦合,提高系統(tǒng)的維護(hù)性和擴(kuò)展性
應(yīng)用場(chǎng)景:有多個(gè)產(chǎn)品類(lèi)時(shí)就要用到工廠模式到旦,比如在數(shù)據(jù)庫(kù)連接中旨巷,我們可以采用多種數(shù)據(jù)庫(kù)連接方法,有mysql擴(kuò)展,mysqli擴(kuò)展,PDO擴(kuò)展等旅掂,在這種情況下我們可以一個(gè)擴(kuò)展對(duì)應(yīng)一個(gè)產(chǎn)品類(lèi),然后采用工廠模式斧吐。
1.分兩部分,產(chǎn)品類(lèi)和工廠類(lèi)仲器,其中產(chǎn)品類(lèi)有多個(gè)煤率,而工廠類(lèi)只有一個(gè)。
2.工廠類(lèi)必須提供一個(gè)生產(chǎn)產(chǎn)品的方法乏冀。
工廠模式實(shí)現(xiàn)思路:
1蝶糯、工廠模式是針對(duì)“相同模型”的產(chǎn)出。即辆沦,使用工廠模式產(chǎn)出的相應(yīng)對(duì)象類(lèi)都有相似的結(jié)構(gòu)或者功能昼捍。所以识虚,首先就需要一批具有相似功能的類(lèi)(本質(zhì)是大類(lèi)下面的小類(lèi),比如人類(lèi)下的男人妒茬,女人)担锤。
//工廠模式
//1、需要使用的類(lèi)1
class Man{
public function display(){
echo '這是男人';
}
}
//1乍钻、需要使用的類(lèi)2
class Woman{
public function display(){
echo '這是女人';
}
}
//1肛循、需要使用的類(lèi)3
class Ladyboy{
public function display(){
echo '這是人妖';
}
}
2、以前訪問(wèn)這些類(lèi)都需要new類(lèi)名實(shí)現(xiàn)银择。為了集中實(shí)例化多糠,我們引入了一個(gè)HumanFactory(人類(lèi)工廠)類(lèi)。用以專(zhuān)門(mén)對(duì)類(lèi)進(jìn)行實(shí)例化浩考。
知識(shí)點(diǎn):
類(lèi)new 一個(gè)字符串熬丧,則是直接實(shí)例化字符串相同名的類(lèi)。我們用變量代替字符串怀挠,就可以實(shí)現(xiàn)動(dòng)態(tài)實(shí)例化類(lèi)的效果。
由此我們的簡(jiǎn)單工廠模式害捕,就設(shè)計(jì)完成绿淋。
簡(jiǎn)單工廠初始:
<?php
//工廠模式
//1、需要使用的類(lèi)1
class Man{
public function display(){
echo '這是男人';
}
}
//1尝盼、需要使用的類(lèi)2
class Woman{
public function display(){
echo '這是女人';
}
}
//1吞滞、需要使用的類(lèi)3
class Ladyboy{
public function display(){
echo '這是人妖';
}
}
//2、工廠類(lèi) HumanFactory
class HumanFactory{
//工廠方法:專(zhuān)門(mén)產(chǎn)生類(lèi)的對(duì)象
public function geyInstance($classname){
return new $classname;
}
}
// 使用公共
$hf = new HumanFactory();
//男人
$m = $hf->geyInstance('Man');
$m->display();
3盾沫、上述工廠類(lèi)在對(duì)象生產(chǎn)的時(shí)候裁赠,額外產(chǎn)生了個(gè)工廠類(lèi)的對(duì)象,實(shí)際上毫無(wú)意義赴精。因此可以使用更優(yōu)的方式來(lái)產(chǎn)生對(duì)象:靜態(tài)工廠佩捞。
靜態(tài)工廠
<?php
//靜態(tài)工廠模式
//1、需要使用的類(lèi)1
class Man{
static public function display(){
echo '這是男人';
}
}
//1蕾哟、需要使用的類(lèi)2
class Woman{
static public function display(){
echo '這是女人';
}
}
//1一忱、需要使用的類(lèi)3
class Ladyboy{
static public function display(){
echo '這是人妖';
}
}
//2、工廠類(lèi) HumanFactory
//3谭确、靜態(tài)工廠類(lèi)
class HumanFactory{
//工廠方法:專(zhuān)門(mén)產(chǎn)生類(lèi)的對(duì)象
static public function geyInstance($classname){
return new $classname;
}
}
//靜態(tài)工廠使用
$m = HumanFactory::geyInstance('Man');
$m::display();
4帘营、以上模式雖然是工廠生產(chǎn)對(duì)象,但是建立在提前知道類(lèi)名的情況下的逐哈。如果類(lèi)名修改芬迄,依然需要修改多處地方。沒(méi)有達(dá)到工廠模式真正的目的昂秃,所以還需要改進(jìn)禀梳。
為了解決類(lèi)名必須知道的問(wèn)題杜窄。我們可以?xún)?nèi)部封裝對(duì)應(yīng)工廠名,這個(gè)模式我們稱(chēng)之為:匿名工廠
匿名工廠
<?php
//靜態(tài)工廠模式
//1出皇、需要使用的類(lèi)1
class Man{
static public function display(){
echo '這是男人';
}
}
//1羞芍、需要使用的類(lèi)2
class Woman{
static public function display(){
echo '這是女人';
}
}
//1、需要使用的類(lèi)3
class Ladyboy{
static public function display(){
echo '這是人妖';
}
}
//2郊艘、工廠類(lèi) HumanFactory
//3荷科、匿名工廠類(lèi)
class HumanFactory{
/**
* @param $flag 封裝的標(biāo)志名
*
*/
static public function geyInstance($flag){
switch ($flag){
case 'm':
return new Man();
case 'w':
return new Woman();
case 'l':
return new Ladyboy();
default:
return null;
}
}
}
//匿名工廠類(lèi)使用
$m = HumanFactory::geyInstance('m');
$m::display();
知識(shí)點(diǎn)
:return 是直接返回值,然后結(jié)束函數(shù)纱注∥方可以不需要break
當(dāng)類(lèi)名被封裝后,如何類(lèi)被修改狞贱。則指修改new 的對(duì)應(yīng)類(lèi)名就可以了刻获。而被調(diào)用的標(biāo)志名可以不變。至此瞎嬉,我們想要達(dá)到的效果完成蝎毡。
5、一般產(chǎn)品類(lèi)就一個(gè)方法氧枣,由此我們可以增加個(gè)魔術(shù)方法沐兵,用自動(dòng)調(diào)用增加代碼效率。
靜態(tài)工廠優(yōu)化版
<?php
//靜態(tài)工廠模式
//1便监、需要使用的類(lèi)1
class Man{
public function __construct()
{
echo '這是男人';
}
}
//1扎谎、需要使用的類(lèi)2
class Woman{
public function __construct()
{
echo '這是女人';
}
}
//1、需要使用的類(lèi)3
class Ladyboy{
public function __construct()
{
echo '這是人妖';
}
}
//2烧董、工廠類(lèi) HumanFactory
//3毁靶、匿名工廠類(lèi)
class HumanFactory{
/**
* @param $flag 封裝的標(biāo)志名
*
*/
static public function geyInstance($flag){
switch ($flag){
case 'm':
return new Man();
case 'w':
return new Woman();
case 'l':
return new Ladyboy();
default:
return null;
}
}
}
//匿名工廠類(lèi)使用
$m = HumanFactory::geyInstance('m');
6、為了給對(duì)應(yīng)類(lèi)約束逊移,我們一般會(huì)在產(chǎn)品類(lèi)之前添加一個(gè)接口预吆。所以標(biāo)準(zhǔn)版的簡(jiǎn)單工廠。如下
標(biāo)準(zhǔn)簡(jiǎn)單工廠
<?php
interface FunctionFactory{
public function Human();
}
//靜態(tài)工廠模式
//1螟左、需要使用的類(lèi)1
class Man implements FunctionFactory {
public function Human()
{
echo '這是男人';
}
public function __construct()
{
$this->Human();
}
}
//1啡浊、需要使用的類(lèi)2
class Woman implements FunctionFactory{
public function Human()
{
echo '這是女人';
}
public function __construct()
{
$this->Human();
}
}
//1、需要使用的類(lèi)3
class Ladyboy implements FunctionFactory{
public function Human()
{
echo '這是人妖';
}
public function __construct()
{
$this->Human();
}
}
//2胶背、工廠類(lèi) HumanFactory
//3巷嚣、匿名工廠類(lèi)
class HumanFactory{
/**
* @param $flag 封裝的標(biāo)志名
*
*/
static public function geyInstance($flag){
switch ($flag){
case 'm':
return new Man();
case 'w':
return new Woman();
case 'l':
return new Ladyboy();
default:
return null;
}
}
}
//匿名工廠類(lèi)使用
$m = HumanFactory::geyInstance('m');
總結(jié):
1、簡(jiǎn)單工廠模式是一種按需生產(chǎn)對(duì)象的方式
2钳吟、簡(jiǎn)單工廠模式一般主要用在大型項(xiàng)目中廷粒,此種項(xiàng)目里面會(huì)出現(xiàn)很多相同的類(lèi),由此需要工廠模式產(chǎn)生對(duì)象。
3坝茎、簡(jiǎn)單工廠模式的優(yōu)點(diǎn)在于后期維護(hù)(修改類(lèi)名)
4涤姊、簡(jiǎn)單工廠模式的缺點(diǎn)在于,隨著功能增加會(huì)增加開(kāi)發(fā)量(開(kāi)發(fā)多個(gè)工廠)
5嗤放、相對(duì)于工廠方法而言思喊,代碼更顯輕量。時(shí)候小型項(xiàng)目使用次酌。
工廠方法
簡(jiǎn)單工廠最大的問(wèn)題在于恨课,如果要添加新功能,就需要修改原類(lèi)岳服,原有方法剂公。這對(duì)于代碼的靈活支持不友好。不符合設(shè)計(jì)模式的開(kāi)閉原則吊宋。嚴(yán)格來(lái)講纲辽,簡(jiǎn)單工廠不是一個(gè)真正的設(shè)計(jì)模式。
為了實(shí)現(xiàn)代碼的擴(kuò)展璃搜,我們把各個(gè)類(lèi)給分別創(chuàng)建工廠拖吼。實(shí)現(xiàn)調(diào)用對(duì)應(yīng)工廠就可以了了。
工廠方法模式
<?php
//抽象工廠 依舊上簡(jiǎn)單工廠 創(chuàng)建工廠的類(lèi)
interface Car{
public function name();
}
//工廠方法模式
// 具體工廠 1 寶馬
class BMW implements Car {
public function name()
{
// 100行初始化代碼
echo '寶馬';
}
public function __construct()
{
$this->name();
}
}
// 具體工廠2 奔馳
class Benz implements Car{
public function name()
{
// 100行初始化代碼
echo '奔馳';
}
public function __construct()
{
$this->name();
}
}
// 具體工廠3 比亞迪
class BYD implements Car{
public function name()
{
// 100行初始化代碼
echo '比亞迪';
}
public function __construct()
{
$this->name();
}
}
//與上面簡(jiǎn)單工廠模式對(duì)比这吻。這里本質(zhì)區(qū)別在于绿贞,此處是將對(duì)象的創(chuàng)建抽象成一個(gè)接口。(也就是把匿名工廠類(lèi)替換成了產(chǎn)品接口)
// 抽象產(chǎn)品 創(chuàng)建實(shí)現(xiàn)產(chǎn)品的接口 ; 需要使用時(shí) 提供對(duì)應(yīng)的產(chǎn)品接口 就完成了實(shí)現(xiàn); 從而達(dá)到了 去 if else 目的
interface CarFactory{
static public function getCar();
}
// 具體產(chǎn)品1 寶馬
class BWMFactory implements CarFactory{
static public function getCar()
{
// TODO: Implement getCar() method.
return new BMW();
}
}
//具體產(chǎn)品 2 奔馳
class BenzFactory implements CarFactory{
static public function getCar()
{
// TODO: Implement getCar() method.
return new Benz();
}
}
// 具體產(chǎn)品2 比亞迪
class BYDFactory implements CarFactory{
static public function getCar()
{
// TODO: Implement getCar() method.
return new BYD();
}
}
// ===========================新增工廠 場(chǎng)景 ====================================
//新增法拉利 工廠
class Ferrari implements Car{
public function name()
{
echo '法拉利';
}
public function __construct()
{
$this->name();
}
}
// 新增法拉利工廠方法
class FerrariFactory implements CarFactory{
static public function getCar()
{
// TODO: Implement getCar() method.
return new Ferrari();
}
}
// ================= 執(zhí)行場(chǎng)景 =========================
// 使用奔馳
BenzFactory::getCar();
// 使用法拉利
FerrariFactory::getCar();
總結(jié)
1橘原、工廠方法,在原來(lái)簡(jiǎn)單工廠的層面上降低了耦合涡上,增加了可擴(kuò)展性
2趾断、但工廠方法相對(duì)簡(jiǎn)單來(lái)說(shuō),代碼復(fù)雜度增加吩愧,編程復(fù)雜度也增加芋酌。
3、由于上面特點(diǎn)雁佳,工廠方法脐帝,更適合用在更為復(fù)雜的大中型項(xiàng)目中。
抽象工廠
抽象工廠糖权,就是在原來(lái)產(chǎn)品工廠類(lèi)的基礎(chǔ)上堵腹,再增加個(gè)工廠的分發(fā)工廠類(lèi)(也就是所謂超級(jí)工廠)。實(shí)現(xiàn)工廠的工廠目的星澳。
抽象工廠模式的組成
抽象工廠(AbstractFactory):確定工廠的業(yè)務(wù)范圍疚顷。
具體工廠(ConcreteFactory):每個(gè)具體工廠對(duì)應(yīng)一個(gè)產(chǎn)品族。具體工廠決定生產(chǎn)哪個(gè)具體產(chǎn)品對(duì)象。
抽象產(chǎn)品(AbstractProduct):同一產(chǎn)品等級(jí)結(jié)構(gòu)的抽象類(lèi)腿堤。
具體產(chǎn)品(ConcreteProduct):可供生產(chǎn)的具體產(chǎn)品阀坏。
抽象工廠實(shí)現(xiàn)
產(chǎn)品類(lèi)
//產(chǎn)品類(lèi) 增加不同種類(lèi)(汽車(chē),空調(diào))的不同產(chǎn)品(【奔馳笆檀、奧迪】【格力忌堂、海爾】)
// 汽車(chē)(抽象產(chǎn)品接口)
interface AutoProduct
{
public function dirve();
}
//奧迪A4(具體產(chǎn)品類(lèi))
class AudiA4Product implements AutoProduct
{
//獲取汽車(chē)名稱(chēng)
public function dirve()
{
// 100行初始化代碼
echo "開(kāi)奧迪A4"."<br>";
}
}
//奔馳C200(具體產(chǎn)品類(lèi))
class BenzC200Product implements AutoProduct
{
//獲取汽車(chē)名稱(chēng)
public function dirve()
{
// 100行初始化代碼
echo "開(kāi)奔馳C200"."<br>";
}
}
//空調(diào)(抽象產(chǎn)品接口)
interface AirCondition
{
public function blow();
}
//格力空調(diào)某型號(hào)(具體產(chǎn)品類(lèi))
class GreeAirCondition implements AirCondition
{
public function blow()
{
echo "吹格力空調(diào)某型號(hào)"."<br>";
}
}
//海爾空調(diào)某型號(hào)(具體產(chǎn)品類(lèi))
class HaierAirCondition implements AirCondition
{
public function blow()
{
echo "吹海爾空調(diào)某型號(hào)"."<br>";
}
}
工廠類(lèi)
//工廠接口
//定義車(chē)的功能,能開(kāi)酗洒,能吹空調(diào)
interface Factory
{
public function getAuto();
public function getAirCondition();
}
//實(shí)現(xiàn) 具體 產(chǎn)品族的組合
//工廠A = 奧迪A4 + 海爾空調(diào)某型號(hào)
class AFactory implements Factory
{
//汽車(chē)
public function getAuto()
{
return new AudiA4Product();
}
//空調(diào)
public function getAirCondition()
{
return new HaierAirCondition();
}
}
//工廠B = 奔馳C200 + 格力空調(diào)某型號(hào)
class BFactory implements Factory
{
//汽車(chē)
public function getAuto()
{
return new BenzC200Product();
}
//空調(diào)
public function getAirCondition()
{
return new GreeAirCondition();
}
}
客戶(hù)端類(lèi)
<?php
//客戶(hù)端測(cè)試代碼
//A工廠制作車(chē)
$factoryA = new AFactory();
$auto_carA = $factoryA->getAuto();
$auto_airA = $factoryA->getAirCondition();
//B工廠制作車(chē)
$factoryB = new BFactory();
$auto_carB = $factoryB->getAuto();
$auto_airB = $factoryB->getAirCondition();
//開(kāi)奧迪車(chē)+吹海爾空調(diào)
$auto_carA->dirve();
$auto_airA->blow(); //熱的時(shí)候可以吹吹空調(diào)
//奔馳C200 + 格力空調(diào)某型號(hào)
$auto_carB->dirve();
$auto_airB->blow(); //熱的時(shí)候可以吹吹空調(diào)