簡單記錄設計模式01(單例、工廠)

可以這樣說吧,你設計出來的東西,是以后給別人用的,當然包括自己。比如,可能你的東西作的很牛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();
最后編輯于
?著作權歸作者所有,轉載或內容合作請聯系作者
平臺聲明:文章內容(如有圖片或視頻亦包括在內)由作者上傳并發布,文章內容僅代表作者本人觀點,簡書系信息發布平臺,僅提供信息存儲服務。

推薦閱讀更多精彩內容