相信很多朋友利用PHP在进行Web开发时都会用到设计模式,其中单例模式应该是应用最多的模式之一,本文并不讨论PHP的各种设计模式,而是重点和大家一起来分析、探讨、分享我自己对PHP单例模式的通俗认识和理解,旨在让大家彻底认清PHP单例模式的本质,我会用最通俗的语言把抽象问题具体化和通俗化,希望通过我的分享能让大家真正了解并掌握PHP单例模式。
最后我会简单的说一下PHP单例模式的缺点。
同时由于本人才疏学浅,对问题认知难免有偏差,本着学习与共享的精神和大家一起探讨,若有不对之处,望大家多多批评指正。
好了,言归正传,让我们一步步来揭开PHP单例模式的神秘面纱……。
What–什么是单例模式呢?
首先我们明确单例模式这个概念,正如其名称一样,单例模式是指整个应用(呵呵,你可能会问什么应用阿?还是有些抽象,别急,下面我们会举例说明)中类只有一个对象实例的的设计模式。
Why–为什么要使用PHP单例模式?
PHP的一个主要应用场合就是应用程序与数据库打交道的应用场景,所以一个应用中会存在大量的数据库操作,比如过数据库句柄来连接数据库这一行为,使用单例模式可以避免大量的new操作,因为每一次new操作都会消耗内存资源和系统资源。
还是有些抽象,给出代码片段。
使用传统方式编码
01 |
<?php |
02 |
...... |
03 |
//初始化一个数据库句柄 |
04 |
$db = new DB(...); |
05 |
//比如有个应用场景是添加一条用户信息: |
06 |
$db ->addUserInfo(); |
07 |
...... |
08 |
//然而我们在另外一个地方可能要查找用户的信息,这个情景出现在一个函数中,这时要用到数据库句柄资源,我们可能需要这么去做 |
09 |
...... |
10 |
function test(){ |
11 |
...... |
12 |
//这时我们不得不重新初始化一个数据库句柄,试想多个应用场景下,这样的代码是多么可怕啊?! |
13 |
$db = new DB(...); |
14 |
$db ->getUserInfo(); |
15 |
...... |
16 |
//有些朋友或许会说,我也可以不这样做啊,我直接利用global关键字不就可以了吗?的确,global可以解决问题,也起到了单例模式的作用,但是OOP中,我们拒绝这样来编写代码,因为global存在安全隐患,请参考相关书籍,同时单例模式恰恰是对全局变量的一种改进,避免了那些存储唯一实例的全局变量污染命名空间 |
17 |
global $db ; //OOP中,我们不提倡这样编写代码 |
18 |
...... |
19 |
} |
使用单例模式编码
1 |
<?php |
2 |
...... |
3 |
//所有的应用情景只有一个数据库句柄资源,嘿嘿,效率老高了, |
4 |
//资源也大大的得到节省,代码简洁明了:) |
5 |
DB::getInstance()->addUserInfo(); |
6 |
DB::getInstance()->getUserInfo(); |
7 |
...... |
How–如何来编写PHP单例模式?
在了解了单例模式的应用场景之后,下面我们通过编写单例模式的具体实现代码来掌握PHP单例模式的核心要点,代码如下:
01 |
<?php |
02 |
/** |
03 |
* PHP单例模式演示举例 |
04 |
* @author guohua.li |
05 |
* @modify 2010-07-11 |
06 |
* @website http://blog.163.com/lgh_2002/ |
07 |
*/ |
08 |
class User{ |
09 |
/** |
10 |
* 静态成品变量 保存全局实例 |
11 |
* @access private |
12 |
*/ |
13 |
static private $_instance = NULL; |
14 |
/** |
15 |
* 私有化构造函数,防止外界实例化对象 |
16 |
*/ |
17 |
private function __construct() {} |
18 |
/** |
19 |
* 私有化克隆函数,防止外界克隆对象 |
20 |
*/ |
21 |
private function __clone(){} |
22 |
/** |
23 |
* 静态方法, 单例统一访问入口 |
24 |
* @return object 返回对象的唯一实例 |
25 |
*/ |
26 |
static public function getInstance() { |
27 |
if ( is_null (self:: $_instance ) || !isset(self:: $_instance )) { |
28 |
self:: $_instance = new self(); |
29 |
} |
30 |
return self:: $_instance ; |
31 |
} |
32 |
/** |
33 |
* 测试方法: 获取用户名字 |
34 |
*/ |
35 |
public function getName() { |
36 |
echo 'hello liguohua!' ; |
37 |
} |
38 |
} |
从以上代码中,我们总结出PHP单例模式实现的核心要点有如下三条:
1. 需要一个保存类的唯一实例的静态成员变量(通常为$_instance私有变量)
2. 构造函数和克隆函数必须声明为私有的,这是为了防止外部程序new类从而失去单例模式的意义
3. 必须提供一个访问这个实例的公共的静态方法(通常为getInstance方法),从而返回唯一实例的一个引用
PHP单例模式的缺点
众所周知,PHP语言是一种解释型的脚本语言,这种运行机制使得每个PHP页面被解释执行后,所有的相关资源都会被回收。也就是说,PHP在语言级别上没有办法让某个对象常驻内存,这和asp.net、Java等编译型是不同的,比如在Java中单例会一直存在于整个应用程序的生命周期里,变量是跨页面级的,真正可以做到这个实例在应用程序生命周期中的唯一性。然而在PHP中,所有的变量无论是全局变量还是类的静态成员,都是页面级的,每次页面被执行时,都会重新建立新的对象,都会在页面执行完毕后被清空,这样似乎PHP单例模式就没有什么意义了,所以PHP单例模式我觉得只是针对单次页面级请求时出现多个应用场景并需要共享同一对象资源时是非常有意义的。
转载于:https://www.cnblogs.com/in-loading/archive/2011/09/20/2182949.html