鸡肋的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截取中文字符串的问题
Jul 12 PHP
十天学会php之第八天
Oct 09 PHP
PHP中for循环语句的几种变型
Mar 16 PHP
discuz7 phpMysql操作类
Jun 21 PHP
php采集时被封ip的解决方法
Aug 29 PHP
php 魔术常量详解及实例代码
Dec 04 PHP
浅谈PHP的$_SERVER[SERVER_NAME]
Feb 04 PHP
Laravel 的数据库迁移的方法
Jul 31 PHP
PHP迭代器接口Iterator用法分析
Dec 28 PHP
PHP图像处理技术实例总结【绘图、水印、验证码、图像压缩】
Dec 08 PHP
php快速导入大量数据的实例方法
Sep 23 PHP
Thinkphp 框架基础之入口文件功能、定义与用法分析
Apr 27 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显示MySQL数据的三种方法
2008/06/05 PHP
CentOS 6.2使用yum安装LAMP以及phpMyadmin详解
2013/06/17 PHP
Drupal7连接多个数据库及常见问题解决
2014/03/02 PHP
php字符串过滤与替换小结
2015/01/26 PHP
memcache一致性hash的php实现方法
2015/03/05 PHP
优化WordPress的Google字体以加速国内服务器上的运行
2015/11/24 PHP
javascipt基础内容--需要注意的细节
2013/04/10 Javascript
jquery统计用户选中的复选框的个数
2014/06/06 Javascript
JQuery1.8 判断元素是否绑定事件的方法
2014/07/10 Javascript
浏览器环境下JavaScript脚本加载与执行探析之defer与async特性
2016/01/14 Javascript
ECMAScript6轮播图实践知识总结
2016/08/17 Javascript
JavaScript自定义分页样式
2017/01/17 Javascript
form表单序列化详解(推荐)
2017/08/15 Javascript
微信小程序实现topBar底部选择栏效果
2018/07/20 Javascript
了解javascript中let和var及const关键字的区别
2019/05/24 Javascript
[35:43]2018DOTA2亚洲邀请赛 4.1 小组赛B组 paiN vs Effect
2018/04/03 DOTA
用Python编写一个基于终端的实现翻译的脚本
2015/04/24 Python
Python实现简单登录验证
2016/04/13 Python
python 开发的三种运行模式详细介绍
2017/01/18 Python
Django 实现下载文件功能的示例
2018/03/06 Python
Flask核心机制之上下文源码剖析
2018/12/25 Python
查看python安装路径及pip安装的包列表及路径
2019/04/03 Python
Django 对象关系映射(ORM)源码详解
2019/08/06 Python
python 画出使用分类器得到的决策边界
2019/08/21 Python
使用PyInstaller将Pygame库编写的小游戏程序打包为exe文件及出现问题解决方法
2019/09/06 Python
python带参数打包exe及调用方式
2019/12/21 Python
python给图像加上mask,并提取mask区域实例
2020/01/19 Python
python模式 工厂模式原理及实例详解
2020/02/11 Python
纯CSS3实现绘制各种图形实现代码详细整理
2012/12/26 HTML / CSS
某公司C#程序员面试题笔试题
2014/05/26 面试题
介绍一下Python中webbrowser的用法
2013/05/07 面试题
小学班干部竞选演讲稿
2014/04/24 职场文书
个人德育工作总结
2015/03/05 职场文书
微观世界观后感
2015/06/10 职场文书
结婚喜宴祝酒词
2015/08/10 职场文书
大学迎新生欢迎词
2015/09/29 职场文书