可以這樣說吧,你設(shè)計(jì)出來的東西径荔,是以后給別人用的,當(dāng)然包括自己猖凛。比如绪穆,可能你的東西作的很牛X,要開源了玖院,要被收錄為一個(gè)擴(kuò)展包了菠红,此時(shí)就必須用到設(shè)計(jì)模式,讓你的東西暴露出干凈漂亮易用的接口难菌,讓別人去使用试溯,復(fù)雜的東西都隱藏在設(shè)計(jì)模式里。
1郊酒、單例模式
一個(gè)類只能有一個(gè)實(shí)例遇绞,可用來當(dāng)作高級(jí)的靜態(tài)常量,以供共享數(shù)據(jù)用燎窘。
比如Laravel中的Request摹闽,每次請(qǐng)求,里面一大堆數(shù)據(jù)褐健,到處都可以用付鹿。
以下例子,假設(shè)有一個(gè)用于保存應(yīng)用程序信息的Preferences類蚜迅,里面保存一些URL根目錄舵匾、文件路徑等數(shù)據(jù)。對(duì)于每一個(gè)應(yīng)用程序谁不,該類應(yīng)該都只有一個(gè)實(shí)例坐梯,確保任何調(diào)用他的地方,里面的數(shù)據(jù)都是一樣的刹帕。在某處更改了烛缔,其他地方立馬可以得到新的數(shù)據(jù)。
<?php
class Preferences
{
private $props = [];
private static $instance;
private function __construct()
{
// 把構(gòu)造函數(shù)設(shè)為private轩拨,所以無法從外部實(shí)例化
}
public static function getInstance()
{
if (empty(self::$instance)) {
self::$instance = new Preferences();
}
return self::$instance;
}
public function __set($key, $val)
{
$this->props[$key] = $val;
}
public function __get($key)
{
return $this->props[$key];
}
}
2践瓷、工廠模式
簡(jiǎn)單來說,就是用一個(gè)工廠類亡蓉,通過調(diào)用工廠類的方法晕翠,來得到一個(gè)產(chǎn)品類的實(shí)例。
為什么不直接用new 產(chǎn)品類()
來得到實(shí)例呢?
可不可以這樣理解淋肾,比如我(客戶硫麻、使用者)要什么東西,打算去工廠(工廠類)樊卓,直接跟廠長(zhǎng)(工廠類的一個(gè)方法)說拿愧,我要圓珠筆(一個(gè)產(chǎn)品類的標(biāo)志,簡(jiǎn)單好記碌尔,而真正的產(chǎn)品類名可能是XX-YY-01這樣的代號(hào))浇辜,然后工廠就會(huì)給你制造一個(gè)圓珠筆(return new XX-YY-01()
)出來。而客戶自己不需要用生產(chǎn)產(chǎn)品的機(jī)器(產(chǎn)品類)來制作(new)圓珠筆(實(shí)例)唾戚。而且柳洋,如果以后工廠改進(jìn)了該圓珠筆,為了區(qū)分叹坦,圓珠筆代號(hào)(產(chǎn)品類名)改成了XX-YY-02熊镣,如果客戶自己生產(chǎn)募书,則還是生產(chǎn)的老版本圓珠筆。而去工廠跟廠長(zhǎng)說來一個(gè)圓珠筆毕箍,此時(shí)得到的將是XX-YY-02代號(hào)的新型圓珠筆而柑,而客戶根本不需要知道這些荷逞。
工廠模式分為:簡(jiǎn)單工廠方法模式、抽象工廠模式和原型模式
讓我們來擴(kuò)展下上面的工廠涩澡,為了擴(kuò)展業(yè)務(wù)妙同,該圓珠筆工廠改為工廠膝迎,而圓珠筆工廠變?yōu)閳A珠筆制造分廠,并新增了鋼筆制造分廠芒涡。
工廠方法模式:圓珠筆工廠只生產(chǎn)一種圓珠筆,鋼筆工廠也是赠群。
<?php
abstract class BasePenFactory
{
abstract public function getPen();
}
class BallpenFactory extends BasePenFactory
{
public function getPen()
{
return new BallpenV01();
}
}
abstract class BasePen
{
abstract public function write();
}
class BallpenV01 extends BasePen
{
public function write()
{
echo "這是用圓珠筆寫的\n";
}
}
// 鋼筆的類似查描,不寫了
$factory = new BallpenFactory();
// 所以柏卤,客戶不需要知道我們的圓珠筆代號(hào)(類名)是啥,只要“拿筆來”getPen()就行了
$ballpen = $factory->getPen();
$ballpen->write();
抽象工廠模式:但是為了擴(kuò)展產(chǎn)品的多樣性,每個(gè)制造分廠忙灼,又可以制造多種不同型號(hào)的筆钝侠,比如0.2帅韧、0.5的圓珠筆等。
<?php
abstract class BasePenFactory
{
abstract public function getPen($thickness);
}
class BallpenFactory extends BasePenFactory
{
public function getPen($thickness)
{
switch ($thickness) {
case "0.2":
return new Ballpen0_2mmV01();
break;
case "0.5":
return new Ballpen0_5mmV01();
break;
default:
echo "沒有此產(chǎn)品\n";
}
}
}
abstract class BasePen
{
abstract public function write();
}
class Ballpen0_5mmV01 extends BasePen
{
public function write()
{
echo "這是用0.5mm圓珠筆寫的\n";
}
}
class Ballpen0_2mmV01 extends BasePen
{
public function write()
{
echo "這是用0.2mm圓珠筆寫的\n";
}
}
// 鋼筆的類似双妨,不寫了
$factory = new BallpenFactory();
// 所以刁品,客戶不需要知道我們的圓珠筆代號(hào)(類名)是啥挑随,
// 只要“拿0.2mm的筆來”getPen("0.2")就行了
$ballpen = $factory->getPen("0.2");
$ballpen->write();
原型模式:這個(gè)就“高科技”了勒叠,該工廠創(chuàng)建時(shí),里面就2臺(tái)機(jī)器拌汇,一臺(tái)鋼筆克隆機(jī)担猛, 一臺(tái)圓珠筆克隆機(jī),沒有產(chǎn)品先改。當(dāng)一個(gè)客戶需要采購(gòu)0.2的圓珠筆蒸走,和0.5的鋼筆時(shí),給工廠帶去兩種筆的樣品各一個(gè)该溯,以后客戶來采購(gòu)時(shí)狈茉,工廠直接要哪種就給克隆哪種!這樣掸掸,就不需要鋼筆分廠,圓珠筆分廠扰付,或者以后的毛筆分廠,馬克筆分廠了实昨!以后要加新產(chǎn)品盐固,就直接加一個(gè)克隆機(jī)刁卜。
<?php
class PenCloneFactory
{
private $ballpen;
private $pen;
public function __construct(Ballpen $ballpen, Pen $pen)
{
$this->ballpen = $ballpen;
$this->pen = $pen;
}
public function getPen()
{
return clone $this->pen;
}
public function getBallpen()
{
return clone $this->ballpen;
}
}
abstract class BasePen
{
abstract public function write();
}
abstract class Ballpen extends BasePen
{
}
class Ballpen0_5mmV01 extends Ballpen
{
public function write()
{
echo "這是用0.5mm圓珠筆寫的\n";
}
}
class Ballpen0_2mmV01 extends Ballpen
{
public function write()
{
echo "這是用0.2mm圓珠筆寫的\n";
}
}
abstract class Pen extends BasePen
{
}
class Pen0_5mmV01 extends Pen
{
public function write()
{
echo "這是用0.5mm圓珠筆寫的\n";
}
}
class Pen0_2mmV01 extends Pen
{
public function write()
{
echo "這是用0.2mm圓珠筆寫的\n";
}
}
// 客戶提交需求长酗,并自帶樣品,工廠把樣品存起來
$factory = new PenCloneFactory(new Ballpen0_2mmV01(), new Pen0_5mmV01());
// 工廠開始根據(jù)客戶的采購(gòu)之拨,克隆之前客戶交代的產(chǎn)品類型
$ballpen = $factory->getBallpen();
$pen = $factory->getPen();
$ballpen->write();
$pen->write();