可以这样说吧,你设计出来的东西,是以后给别人用的,当然包括自己。比如,可能你的东西作的很牛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();