单例模式(Singleton Pattern):顾名思义,就是只有一个实例。作为对象的创建模式,单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例。
单例模式也细分为懒汉模式和饿汉模式,感兴趣的朋友可以去了解一下。这里阐述的代码实现是懒汉模式
(一)为什么要使用PHP单例模式
1,开发中有些时候,一个应用中会存在大量的数据库操作。 在使用面向对象的方式开发时, 如果使用单例模式,则可以避免大量的new 操作消耗的资源,还可以减少数据库连接这样就不容易出现 too many connections情况。
2,如果系统中需要有一个类来全局控制某些配置信息, 那么使用单例模式可以很方便的实现. 这个可以参看zend Framework的FrontController部分。
3,在一次页面请求中, 便于进行调试, 因为所有的代码(例如数据库操作类db)都集中在一个类中, 我们可以在类中设置钩子函数, 输出日志,从而避免到处var_dump, echo
(二)单例模式的实现
1,私有化一个属性用于存放唯一的一个实例
2,私有化构造方法,私有化克隆方法,用来创建并只允许创建一个实例
3,公有化静态方法,用于向系统提供这个实例
(三)代码实现
class Singleton{
//存放实例
private static $_instance = null;
//私有化构造方法、
private function __construct(){
echo "单例模式的实例被构造了";
}
//私有化克隆方法
private function __clone(){
}
//公有化获取实例方法
public static function getInstance(){
if (!(self::$_instance instanceof Singleton)){
self::$_instance = new Singleton();
}
return self::$_instance;
}
}
$singleton=Singleton::getInstance();
优点:因为静态方法可以在全局范围内被访问,当我们需要一个单例模式的对象时,只需调用getInstance
方法,获取先前实例化的对象,无需重新实例化。
(四)使用Trait关键字实现类似于继承单例类的功能
Trait
是PHP5.4
后加入到特性,有些书说是为了实现类似C++
多重继承的功能。Trait
定义的代码结构和类很相似。又有点像实现逻辑代码的接口。当使用use的时候,Trait
的代码就好像拷贝到use
所在的位置取代use
。从这个角度来看,更像C
的宏
Trait Singleton{
//存放实例
private static $_instance = null;
//私有化克隆方法
private function __clone(){
}
//公有化获取实例方法
public static function getInstance(){
$class = __CLASS__;
if (!(self::$_instance instanceof $class)){
self::$_instance = new $class();
}
return self::$_instance;
}
}
class DB {
private function __construct(){
echo __CLASS__.PHP_EOL;
}
}
class DBhandle extends DB {
use Singleton;
private function __construct(){
echo "单例模式的实例被构造了";
}
}
$handle=DBhandle::getInstance();
//注意若父类方法为public,则子类只能为pubic,
//若父类为private,子类为public ,protected,private都可以。
补充:大多数书籍介绍单例模式,都会讲三私一公,公有化静态方法作为提供对象的接口,私有属性用于存放唯一一个单例对象。私有化构造方法,私有化克隆方法保证只存在一个单例。
但实际上,虽然我们无法通过new
关键字和clone
出一个新的对象,但我们若想得到一个新对象。还是有办法的,那就是通过序列化和反序列化得到一个对象。私有化__sleep()
和__wakeup()
方法依然无法阻止通过这种方法得到一个新对象。或许真得要阻止,你只能去__wakeup添加删除一个实例的代码,保证反序列化增加一个对象,你就删除一个。不过这样貌似有点怪异。