PHP单例模式的本质

    相信很多朋友利用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单例模式我觉得只是针对单次页面级请求时出现多个应用场景并需要共享同一对象资源时是非常有意义的。


=================================================================================


多数 人都是从单例模式的字面上的意思来理解它的用途, 认为这是对系统资源的节省, 可以避免重复实例化, 是一种"计划生育".   而PHP每次执行完页面都是会从内存中清理掉所有的资源. 因而PHP中的单例实际每次运行都是需要重新实例化的, 这样就失去了单例重复实例化的意义了. 单单从这个方面来说, PHP的单例的确有点让各位失望. 但是单例仅仅只有这个功能和应用吗? 答案是否定的,我们一起来看看。

1. php的应用主要在于数据库应用, 所以一个应用中会存在大量的数据库操作, 在使用面向对象的方式开发时(废话), 如果使用单例模式, 则可以避免大量的new 操作消耗的资源。

2. 如果系统中需要有一个类来全局控制某些配置信息, 那么使用单例模式可以很方便的实现. 这个可以参看zend Framework的FrontController部分。

3. 在一次页面请求中, 便于进行调试, 因为所有的代码(例如数据库操作类db)都集中在一个类中, 我们可以在类中设置钩子, 输出日志,从而避免到处var_dump, echo。

 

复制代码
<?php 

/**
* 设计模式之单例模式
* $_instance必须声明为静态的私有变量
* 构造函数和析构函数必须声明为私有,防止外部程序new
* 类从而失去单例模式的意义
* getInstance()方法必须设置为公有的,必须调用此方法
* 以返回实例的一个引用
* ::操作符只能访问静态变量和静态函数
* new对象都会消耗内存
* 使用场景:最常用的地方是数据库连接。 
* 使用单例模式生成一个对象后,
* 该对象可以被其它众多对象所使用。 
*/
class Example
{
//保存例实例在此属性中
private static $_instance;

//构造函数声明为private,防止直接创建对象
private function __construct()
{
echo 'I am Construceted';
}

//单例方法
public static function singleton()
{
if(!isset(self::$_instance))
{
$c=__CLASS__;
self
::$_instance=new $c;
}
return self::$_instance;


//阻止用户复制对象实例
public function __clone()
{
trigger_error('Clone is not allow' ,E_USER_ERROR);
}

function test()
{
echo("test");

}
}

// 这个写法会出错,因为构造方法被声明为private
$test = new Example;

// 下面将得到Example类的单例对象
$test = Example::singleton();
$test->test();

// 复制对象将导致一个E_USER_ERROR.
$test_clone = clone $test;
?>






  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 0
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值