单例模式的要点有三个:
一是某个类只能有一个实例;
二是它必须自行创建这个实例;
三是它必须自行向整个系统提供这个实例。
什么是单例模式:
单例模式就是一个类将直接的构造函数给私有化,禁止外部new实例,而是通过自己内部实现实例化,供外部使用。php最常见的单例模式就是数据库的操作了。
为什么要使用单例:
PHP的一个主要应用场合就是应用程序与数据库打交道的应用场景,所以一个应用中会存在大量的数据库操作,比如过数据库句柄来连接数据库这一行为,使用单例模式可以避免大量的new操作,因为每一次new操作都会消耗内存资源和系统资源。
传统的数据库连接实例:
//初始化一个数据库句柄
$db = new DB(...);
//比如有个应用场景是添加一条用户信息:
$db->addUserInfo();
......
//然而我们在另外一个地方可能要查找用户的信息,这个情景出现在一个函数中,这时要用到数据库句柄资源,我们可能需要这么去做
......
function test(){
......
//这时我们不得不重新初始化一个数据库句柄,试想多个应用场景下,这样的代码是多么可怕啊?!
$db = new DB(...);
$db->getUserInfo();
......
//有些朋友或许会说,我也可以不这样做啊,我直接利用global关键字不就可以了吗?的确,global可以解决问题,也起到了单例模式的作用,但是OOP中,我们拒绝这样来编写代码,因为global存在安全隐患,请参考相关书籍,同时单例模式恰恰是对全局变量的一种改进,避免了那些存储唯一实例的全局变量污染命名空间
global $db; //OOP中,我们不提倡这样编写代码
......
}
使用单例模式编码:
......
//所有的应用情景只有一个数据库句柄资源,嘿嘿,效率老高了,
//资源也大大的得到节省,代码简洁明了:)
DB::getInstance()->addUserInfo();
DB::getInstance()->getUserInfo();
.....
不难发现,我们使用单例时,直接在需要连接数据库的适合直接调用DB类的静态方法就OK了,不需要再new一个数据库的实例了,减少了内存的消耗。
单例模式优点:
因为每一次new操作都会消耗内存资源和系统资源,所以就是类直接在内部实例化自己,避免外部大量的new操作,之后就被外部调用,减少内存消耗。
单例模式缺点:
众所周知,PHP语言是一种解释型的脚本语言,这种运行机制使得每个PHP页面被解释执行后,所有的相关资源都会被回收。也就是说,PHP在语言级别上没有办法让某个对象常驻内存,这和asp.net、Java等编译型是不同的,比如在Java中单例会一直存在于整个应用程序的生命周期里,变量是跨页面级的,真正可以做到这个实例在应用程序生命周期中的唯一性。然而在PHP中,所有的变量无论是全局变量还是类的静态成员,都是页面级的,每次页面被执行时,都会重新建立新的对象,都会在页面执行完毕后被清空,这样似乎PHP单例模式就没有什么意义了,所以PHP单例模式我觉得只是针对单次页面级请求时出现多个应用场景并需要共享同一对象资源时是非常有意义的。
代码的实现主要有以下三点:
1.需要一个保存类的唯一实例的静态成员变量(通常为$_instance私有变量)
2.构造函数和克隆函数必须声明为私有的,这是为了防止外部程序new类从而失去单例模式的意义
3.必须提供一个访问这个实例的公共的静态方法(通常为getInstance方法),从而返回唯一实例的一个引用
单例模式demo:
<?php
class Test{
privite static $_instance;//首先声明一个静态变量,用来保存实例化的类
//将类的构造函数私有化,防止外部new实例化
privite function __construct(){
}
//将类的clone方法也私有化,防止被克隆
privite function __clone(){
}
//创建一个静态方法getInstance,供外部调用
public static function getInstance(){
if(! self::$_instance instanceof self){
self::$_instance = new self;
}
return self::$_instance;
}
//创建一个小实例方法
public function demo(){
echo 'hello,world!';
}
}
$test = Test::getInstance();
$test->demo();
?>