可以這樣說吧,你設計出來的東西,是以后給別人用的,當然包括自己。比如,可能你的東西作的很牛X,要開源了,要被收錄為一個擴展包了,此時就必須用到設計模式,讓你的東西暴露出干凈漂亮易用的接口,讓別人去使用,復雜的東西都隱藏在設計模式里。
1、單例模式
一個類只能有一個實例,可用來當作高級的靜態常量,以供共享數據用。
比如Laravel中的Request,每次請求,里面一大堆數據,到處都可以用。
以下例子,假設有一個用于保存應用程序信息的Preferences類,里面保存一些URL根目錄、文件路徑等數據。對于每一個應用程序,該類應該都只有一個實例,確保任何調用他的地方,里面的數據都是一樣的。在某處更改了,其他地方立馬可以得到新的數據。
<?php
class Preferences
{
private $props = [];
private static $instance;
private function __construct()
{
// 把構造函數設為private,所以無法從外部實例化
}
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、工廠模式
簡單來說,就是用一個工廠類,通過調用工廠類的方法,來得到一個產品類的實例。
為什么不直接用new 產品類()
來得到實例呢?
可不可以這樣理解,比如我(客戶、使用者)要什么東西,打算去工廠(工廠類),直接跟廠長(工廠類的一個方法)說,我要圓珠筆(一個產品類的標志,簡單好記,而真正的產品類名可能是XX-YY-01這樣的代號),然后工廠就會給你制造一個圓珠筆(return new XX-YY-01()
)出來。而客戶自己不需要用生產產品的機器(產品類)來制作(new)圓珠筆(實例)。而且,如果以后工廠改進了該圓珠筆,為了區分,圓珠筆代號(產品類名)改成了XX-YY-02,如果客戶自己生產,則還是生產的老版本圓珠筆。而去工廠跟廠長說來一個圓珠筆,此時得到的將是XX-YY-02代號的新型圓珠筆,而客戶根本不需要知道這些。
工廠模式分為:簡單工廠方法模式、抽象工廠模式和原型模式
讓我們來擴展下上面的工廠,為了擴展業務,該圓珠筆工廠改為工廠,而圓珠筆工廠變為圓珠筆制造分廠,并新增了鋼筆制造分廠。
工廠方法模式:圓珠筆工廠只生產一種圓珠筆,鋼筆工廠也是。
<?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();
// 所以,客戶不需要知道我們的圓珠筆代號(類名)是啥,只要“拿筆來”getPen()就行了
$ballpen = $factory->getPen();
$ballpen->write();
抽象工廠模式:但是為了擴展產品的多樣性,每個制造分廠,又可以制造多種不同型號的筆,比如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 "沒有此產品\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();
// 所以,客戶不需要知道我們的圓珠筆代號(類名)是啥,
// 只要“拿0.2mm的筆來”getPen("0.2")就行了
$ballpen = $factory->getPen("0.2");
$ballpen->write();
原型模式:這個就“高科技”了,該工廠創建時,里面就2臺機器,一臺鋼筆克隆機, 一臺圓珠筆克隆機,沒有產品。當一個客戶需要采購0.2的圓珠筆,和0.5的鋼筆時,給工廠帶去兩種筆的樣品各一個,以后客戶來采購時,工廠直接要哪種就給克隆哪種!這樣,就不需要鋼筆分廠,圓珠筆分廠,或者以后的毛筆分廠,馬克筆分廠了!以后要加新產品,就直接加一個克隆機。
<?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());
// 工廠開始根據客戶的采購,克隆之前客戶交代的產品類型
$ballpen = $factory->getBallpen();
$pen = $factory->getPen();
$ballpen->write();
$pen->write();