鸡肋的PHP单例模式应用详解


Posted in PHP onJune 03, 2013

单例模式的要点有三个:
一是某个类只能有一个实例;
二是它必须自行创建这个实例;
三是它必须自行向整个系统提供这个实例。

<?php 
/* 单例模式举例,其要点如下: 
* 
* 1. $_instance 必须声明为静态的私有变量 
* 2. 构造函数和克隆函数必须声明为私有的,这是为了防止外部程序 new 类从而失去单例模式的意义 
* 3. getInstance()方法必须声明为公有的,必须调用此方法以返回唯一实例的一个引用 
* 4. ::操作符只能访问静态变量或静态函数 
* 5. PHP的单例模式是相对而言的,因为PHP的解释运行机制使得每个PHP页面被解释执行后,所有的相关资源都会被回收。 
* 也就是说,PHP在语言级别上没有办法让某个对象常驻内存。在PHP中,所有的变量都是页面级的,无论是全局变量, 
* 还是类的静态成员,都会在页面执行完毕后被清空,结果会重新建立新的对象,这样也就完全失去了Singleton的意义。 
* 不过,在实际应用中同一个页面中可能会存在多个业务逻辑,这时单例模式就起到了很重要的作用,有效的避免了重复 
* new 对象(注: new 对象会消耗内存资源)这么一个行为,所以我们说PHP的单例模式是相对而言的 
* 
*/ 
class People 
{ 
static private $_instance = NULL; 
public $height = ''; 
public $age = ''; 
private function __construct() 
{ 
$this->height = '185'; 
$this->age = 25; 
} 
private function __clone() 
{ 
//do something 
} 
static public function getInstance() 
{ 
if(!self::$_instance instanceof self) 
{ 
//echo 'lgh-big'; 
self::$_instance = new self; 
} 
else 
{ 
//for testing only 
//echo 'gdc-xiaoairener'; 
} 
return self::$_instance; 
} 
public function getHeight() 
{ 
echo $this->height; 
} 
public function getAge() 
{ 
echo $this->age; 
} 
} 
function testInstance() 
{ 
People::getInstance()->getAge(); 
} 
//begin to use the class 
$lgh = People::getInstance(); 
$lgh->getHeight(); 
echo '<br />'; 
testInstance(); 
?>

优点:单例模式可以避免大量的new操作,因为每一次new操作都会消耗内存资源和系统资源
缺点:在PHP中,所有的变量无论是全局变量还是类的静态成员,都是 页面级的,每次页面被执行时,都会重新建立新的对象,都会在页面执行完毕后被清空,这样似乎PHP单例模式就没有什么意义了,所以PHP单例模式我觉得只 是针对单次页面级请求时出现多个应用场景并需要共享同一对象资源时是非常有意义的。

Why?为什么要使用PHP单例模式?
PHP的一个主要应用场合就是应用程序与数据库打交道的应用场景,所以一个应用中会存在大量的数据库操作,比如过数据库句柄来连接数据库这一行为,使用单例模式可以避免大量的new操作,因为每一次new操作都会消耗内存资源和系统资源。
还是有些抽象,给出代码片段。
使用传统方式编码

<?php
......
//初始化一个数据库句柄
$db = new DB(...);
//比如有个应用场景是添加一条用户信息:
$db->addUserInfo();
......
//然而我们在另外一个地方可能要查找用户的信息,这个情景出现在一个函数中,这时要用到数据库句柄资源,我们可能需要这么去做
......
function test(){
 ......
//这时我们不得不重新初始化一个数据库句柄,试想多个应用场景下,这样的代码是多么可怕啊?!
 $db = new DB(...);
 $db->getUserInfo();
 ......
//有些朋友或许会说,我也可以不这样做啊,我直接利用global关键字不就可以了吗?的确,global可以解决问题,也起到了单例模式的作用,但是OOP中,我们拒绝这样来编写代码,因为global存在安全隐患,请参考相关书籍,同时单例模式恰恰是对全局变量的一种改进,避免了那些存储唯一实例的全局变量污染命名空间
 global $db;  //OOP中,我们不提倡这样编写代码
......
}

使用单例模式编码
<?php
......
//所有的应用情景只有一个数据库句柄资源,嘿嘿,效率老高了,
//资源也大大的得到节省,代码简洁明了:)
DB::getInstance()->addUserInfo();
DB::getInstance()->getUserInfo();
......

How?如何来编写PHP单例模式?
在了解了单例模式的应用场景之后,下面我们通过编写单例模式的具体实现代码来掌握PHP单例模式的核心要点,代码如下:
<?php
/**
 *  PHP单例模式演示举例
*  @author   guohua.li
 *  @modify  2010-07-11
*  @website  http://blog.163.com/lgh_2002/
*/
class User{
/**
 *  静态成品变量 保存全局实例
 *  @access private
*/
static private $_instance = NULL;
/**
 *  私有化构造函数,防止外界实例化对象
*/
private function  __construct() {}
/**
 *  私有化克隆函数,防止外界克隆对象
*/
private function  __clone(){}
/**
 *  静态方法, 单例统一访问入口
 *  @return  object  返回对象的唯一实例
*/
static public function getInstance() {
if (is_null(self::$_instance) || !isset(self::$_instance)) {
self::$_instance = new self();
}
return self::$_instance;
} 
   /**
 * 测试方法: 获取用户名字
*/
public function getName() {
echo 'hello liguohua!';
}
}

从以上代码中,我们总结出PHP单例模式实现的核心要点有如下三条:
1.需要一个保存类的唯一实例的静态成员变量(通常为$_instance私有变量)
2.构造函数和克隆函数必须声明为私有的,这是为了防止外部程序new类从而失去单例模式的意义
3.必须提供一个访问这个实例的公共的静态方法(通常为getInstance方法),从而返回唯一实例的一个引用
PHP单例模式的缺点
众所周知,PHP语言是一种解释型的脚本语言,这种运行机制使得每个PHP页面被解释执行后,所有的相关资源都会被回收。也就是说,PHP在语言级别上没有办法让某个对象常驻内存,这和asp.net、Java等编译型是不同的,比如在Java中单例会一直存在于整个应用程序的生命周期里,变量是跨页面级的,真正可以做到这个实例在应用程序生命周期中的唯一性。然而在PHP中,所有的变量无论是全局变量还是类的静态成员,都是页面级的,每次页面被执行时,都会重新建立新的对象,都会在页面执行完毕后被清空,这样似乎PHP单例模式就没有什么意义了,所以PHP单例模式我觉得只是针对单次页面级请求时出现多个应用场景并需要共享同一对象资源时是非常有意义的。
PHP 相关文章推荐
一段防盗连的PHP代码
Dec 06 PHP
PHP 加密/解密函数 dencrypt(动态密文,带压缩功能,支持中文)
Jan 30 PHP
PHP 截取字符串函数整理(支持gb2312和utf-8)
Feb 16 PHP
PHP目录函数实现创建、读取目录教程实例
Jan 13 PHP
PHP中如何判断AJAX提交的数据
Feb 05 PHP
Php中用PDO查询Mysql来避免SQL注入风险的方法
Apr 25 PHP
php中PDO方式实现数据库的增删改查
May 17 PHP
学习php设计模式 php实现观察者模式(Observer)
Dec 09 PHP
Zend Framework教程之连接数据库并执行增删查的方法(附demo源码下载)
Mar 21 PHP
php使用函数pathinfo()、parse_url()和basename()解析URL
Nov 25 PHP
PHP实现链表的定义与反转功能示例
Jun 09 PHP
php依赖注入知识点详解
Sep 23 PHP
phpize的深入理解
Jun 03 #PHP
PHP不用第三变量交换2个变量的值的解决方法
Jun 02 #PHP
基于php socket(fsockopen)的应用实例分析
Jun 02 #PHP
深入PHP操作MongoDB的技术总结
Jun 02 #PHP
深入php数据采集的详解
Jun 02 #PHP
基于php下载文件的详解
Jun 02 #PHP
用PHP实现浏览器点击下载TXT文档的方法详解
Jun 02 #PHP
You might like
PHP提示Deprecated: mysql_connect(): The mysql extension is deprecated的解决方法
2014/08/28 PHP
分享8个Laravel模型时间戳使用技巧小结
2020/02/12 PHP
增强的 JavaScript 的 trim 函数的代码
2007/08/13 Javascript
javascript css在IE和Firefox中区别分析
2009/02/18 Javascript
javascript CSS画图之基础篇
2009/07/29 Javascript
Javascript倒计时代码
2010/08/12 Javascript
使用js在页面中绘制表格核心代码
2013/09/16 Javascript
js截取固定长度的中英文字符的简单实例
2013/11/22 Javascript
第五章之BootStrap 栅格系统
2016/04/25 Javascript
Bootstrap和Angularjs配合自制弹框的实例代码
2016/08/24 Javascript
js实现表格筛选功能
2017/01/18 Javascript
原生js实现节日时间倒计时功能
2017/01/18 Javascript
Vue组件tree实现树形菜单
2017/04/13 Javascript
Kindeditor单独调用单图上传增加预览功能的实例
2017/07/31 Javascript
js链表操作(实例讲解)
2017/08/29 Javascript
JavaScript原生实现观察者模式的示例
2017/12/15 Javascript
vue几个常用跨域处理方式介绍
2018/02/07 Javascript
基于Vuex无法观察到值变化的解决方法
2018/03/01 Javascript
python获取一组数据里最大值max函数用法实例
2015/05/26 Python
自动化Nginx服务器的反向代理的配置方法
2015/06/28 Python
Python实现定时备份mysql数据库并把备份数据库邮件发送
2018/03/08 Python
padas 生成excel 增加sheet表的实例
2018/12/11 Python
python如何获取当前文件夹下所有文件名详解
2019/01/25 Python
Python 生成器,迭代,yield关键字,send()传参给yield语句操作示例
2019/10/12 Python
django xadmin action兼容自定义model权限教程
2020/03/30 Python
pyqt5中动画的使用详解
2020/04/01 Python
Python docutils文档编译过程方法解析
2020/06/23 Python
Python操作dict时避免出现KeyError的几种解决方法
2020/09/20 Python
Django Model层F,Q对象和聚合函数原理解析
2020/11/12 Python
Python批量删除mysql中千万级大量数据的脚本分享
2020/12/03 Python
基于html5绘制圆形多角图案
2016/04/21 HTML / CSS
中专生毕业自我鉴定
2013/11/01 职场文书
《忆江南》教学反思
2014/04/07 职场文书
新年爱情寄语
2014/04/08 职场文书
竞选团支书演讲稿
2014/04/28 职场文书
推荐信范文大全
2015/03/27 职场文书