简单记录设计模式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();
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
【社区内容提示】社区部分内容疑似由AI辅助生成,浏览时请结合常识与多方信息审慎甄别。
平台声明:文章内容(如有图片或视频亦包括在内)由作者上传并发布,文章内容仅代表作者本人观点,简书系信息发布平台,仅提供信息存储服务。

相关阅读更多精彩内容

友情链接更多精彩内容