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