1)工厂模式

工厂模式是用工厂方法生成对象,而不是直接new一个对象。假设我们在Config命名空间下有一个名叫Db的数据库操作类,用普通的方法,如果我们想去创建一个Db的对象,我们会直接new一个出来。

$db = new \Config\Db();

工厂模式就是用一个工厂方法替换掉直接new一个对象的操作,以后想创建对象就调用这个工厂方法。

<?php 
namespace Config;

class Factory{
    static public function createDb(){
        $db = new Db();
        return $db;
    }
}
$db =   \Config\Factory::createDb();

工厂模式有什么好处呢, 我们的项目中多处都对Db类进行了new的操作,如果这个类发生了一些更改,比如说类名或者是参数的改变,没用工厂模式的话我们就需要进行多处更改,而工厂模式只需要改这个工厂类就行了。

2)单例模式

单例模式使某个类的对象仅能创建一次,通常一个项目中会多次用的Db这个数据库连接类,如果在每个地方都调用工厂方法创建一个数据库连接类,这样是比较消耗资源的,我们只需要一个数据库连接,单例模式就是来解决这个问题的。
我们打开Db类,首先把构造方法设置为私有的,这样就禁止了在其他地方直接new我们的Db类

<?php 
namespace Config;

class Db{
    static protected $db;
    private function __construct(){

    }
    static public function getInstance(){
        //条件判断Db类是否已经new过
        if (self::$db) {
            return self::$db;
        } else {
            //构造方法被设置为了私有的,外部不能直接new,但自己内部可以new
            self::$db = new self();
            return self::$db;
        }
    }

//下面省略一系列操作数据库的方法
}

Db类的构造方法设置为了私有的,那我们在工厂类中也不能直接new了,现在来修改我们的工厂类

<?php 
namespace Config;

class Factory{
    static public function createDb(){
        $db = Db::getInstance;
        return $db;
    }
}

现在不管我们调用多少次工厂方法,我们的数据库连接都只会被创建一次。

3)注册树模式

注册树模式可以把我们的对象放在全局的树上,让对象可以全局共享
下面我们来编写一个注册树的类,包含set,get,_unset三个操作。

<?php 
namespace Config;

class Register{
    protected static $objects;
    static public function set($alias,$object){
        self::$objedts[$alias] = $object;
    }

    static public function get($name){
        return self::$objedts[$name];
    }

    //unset是php中的关键词,所以起名为_unset
    static public function _unset($alias){
        unset(self::$objedts[$alias]);
    }
}

下面我们再把工厂方法改一下

<?php 
namespace Config;

class Factory{
    static public function createDb(){
        $db = Db::getInstance;
        //把单例模式生成的对象放在注册树上
        Register::set('db',$db);
    }
}

现在这个工厂方法只需要调用一次,以后再需要使用数据库连接对象,直接从全局的注册树上拿就行了

 $db = Register::get("db");

至于这个工厂方法是么时候调用,可以在程序初始化的时候,我们的业务逻辑代码只需要在注册树上把这个对象读取出来即可。至此,三种最基本的设计模式就介绍完了。