PHP单例模式(精讲)-程序员宅基地

技术标签: java  php  数据库  

首先我们要明确单例模式这个概念,那么什么是单例模式呢?

单例模式顾名思义,就是只有一个实例。作为对象的创建模式,单例模式确保某一个类只有一个实例,而且自行实例化并向整个系统提供这个实例,这个类我们称之为单例类。

单例模式的要点有三个:

一是某个类只能有一个实例;

二是它必须自行创建这个实例;

三是它必须自行向整个系统提供这个实例。

<?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();
?>

下面我们讨论下为什么要使用PHP单例模式?

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

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

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

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

使用传统方式编码

<?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(); 

PHP单例模式实现的核心要点有如下三条:

1.需要一个保存类的唯一实例的静态成员变量(通常为$_instance私有变量)

2.构造函数和克隆函数必须声明为私有的,这是为了防止外部程序new类从而失去单例模式的意义

3.必须提供一个访问这个实例的公共的静态方法(通常为getInstance方法),从而返回唯一实例的一个引用

在了解了单例模式的应用场景之后,下面我们通过编写单例模式的具体实现代码来掌握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单例模式的缺点

众所周知,PHP语言是一种解释型的脚本语言,这种运行机制使得每个PHP页面被解释执行后,所有的相关资源都会被回收。也就是说,PHP在语言级别上没有办法让某个对象常驻内存,这和asp.net、Java等编译型是不同的,比如在Java中单例会一直存在于整个应用程序的生命周期里,变量是跨页面级的,真正可以做到这个实例在应用程序生命周期中的唯一性。然而在PHP中,所有的变量无论是全局变量还是类的静态成员,都是页面级的,每次页面被执行时,都会重新建立新的对象,都会在页面执行完毕后被清空,这样似乎PHP单例模式就没有什么意义了,所以PHP单例模式我觉得只是针对单次页面级请求时出现多个应用场景并需要共享同一对象资源时是非常有意义的。

参考链接:

http://www.cnblogs.com/zox2011/archive/2011/09/20/2182119.html

转载于:https://my.oschina.net/u/3630946/blog/1519398

版权声明:本文为博主原创文章,遵循 CC 4.0 BY-SA 版权协议,转载请附上原文出处链接和本声明。
本文链接:https://blog.csdn.net/weixin_33766168/article/details/92395284

智能推荐

微信小程序使用getlocation和chooselocation得到的坐标不一致的原因_chooselocation和getlocation-程序员宅基地

文章浏览阅读3k次,点赞3次,收藏4次。这也是一个新手容易犯的错误,技术文档中可以找到wx.getlocation得到的默认坐标系是wgs84即GPS坐标系,但是在chooselocation和openlocation中并未再次提及坐标系type,其实在wx.getlocation中提及过默认返回wgs84,但是可以通过制定type为gcj02用于openlocation,同样在wx.openlocation中的示例也用到了这点:其实只有wx.gentlocation默认返回的是gps系,而其他方法都使用的是火星坐标系也就是gcj02,..._chooselocation和getlocation

fopen和open区别_c语言fopen和fatfs的f_open有什么区别-程序员宅基地

文章浏览阅读952次。1、open 是系统调用 返回的是文件句柄,文件的句柄是文件在文件描述副表里的索引,fopen是C的库函数,返回的是一个指向文件结构的指针。2、fopen的实现要调用open, 关键看你想返回什么了, FILE指针还是描述符? 3、32位环境下,编译加“-D_FILE_OFFSET_BITS=64”要在open里加O_LARGEFILE标记[code]static _c语言fopen和fatfs的f_open有什么区别

Android开发:实时处理摄像头预览帧视频------浅析PreviewCallback,onPreviewFrame,AsyncTask的综合应用_onpreviewframe 中进行人脸对比-程序员宅基地

文章浏览阅读729次。很多时候,android摄像头模块不仅预览,拍照这么简单,而是需要在预览视频的时候,能够做出一些检测,比如最常见的人脸检测。在未按下拍照按钮前,就检测出人脸然后矩形框标示出来,再按拍照。那么如何获得预览帧视频么?只需要在Activity里继承PreviewCallback这个接口就行了。示例如下:public class RectPhoto extends Activity_onpreviewframe 中进行人脸对比

软件工程生涯人物访谈报告_软件工程师访谈报告-程序员宅基地

文章浏览阅读1.6w次,点赞2次,收藏19次。要求:提交纸质材料符合规范格式,封面统一。字体及格式要求如下:文章题目(三号、黑体、加粗,居中)正文内容(小四,宋体,1.5倍行距,段首空2格)不允许抄袭,一经发现以不及格处理。​ 成绩:嘉兴学院课程名称: 《准职业人导向训练1》任课教师:题目:生涯人物访谈报告学院:ICT学院年级: 2019****级专业及班级: 软件****191姓名:学号:​ 提交日期:2019年 12 月****20 日生涯人物访谈报告_软件工程师访谈报告

字节跳动面试必问,Android-NDK开发入门,内含福利-程序员宅基地

文章浏览阅读911次,点赞22次,收藏18次。写到这里也结束了,在文章最后放上一个小小的福利,以下为小编自己在学习过程中整理出的一个关于Flutter的学习思路及方向,从事互联网开发,最主要的是要学好技术,而学习技术是一条慢长而艰苦的道路,不能靠一时激情,也不是熬几天几夜就能学好的,必须养成平时努力学习的习惯,更加需要准确的学习方向达到有效的学习效果。由于内容较多就只放上一个大概的大纲,需要更及详细的学习思维导图的。

小菜鸡手把手教你electron-vue结合python的flask web框架打包成桌面离线应用_python electron vue-程序员宅基地

文章浏览阅读2.8k次。electron-vue是可以用vue开发客户端应用的一个强大工具。简而言之就是能把你的web页面打包成桌面应用。python的flask web框架则是小而好用,做本地服务应该够用了。为啥这样搭配?单纯用electron-vue里的node.js开发服务什么的也可以,不过感觉还是python写后台代码爽,python也有qt5之类的开发客户端界面,但是感觉麻烦,刚好我会vue和python,也只想弄个小工具耍耍,这样搭配刚好满足我的开发需求,接下来我就吹吹水分享下我的搭建过程。electr_python electron vue

随便推点

Centos 7.x 源码安装Ansible参考篇_centos 安装 ansible 国内源-程序员宅基地

文章浏览阅读956次。Ansible架构图_centos 安装 ansible 国内源

ddr3配置 dsp6678_基于Xilinx Virtex6+C6678 DSP+DDR3+ADC的6U OpenVPX信号处理板-程序员宅基地

文章浏览阅读591次。VPX613是一款基于6U OpenVPX总线架构的超高速信号采集、处理、回放板卡,该板卡包括1片单通道5.2Gsps DAC(可选兼容4Gsps版本),1片单通道5Gsps ADC(可配置成2通道2.5G或4通道1.25G),1片用于信号处理的高端Xilinx Virtex6系列FPGA以及一片TI的C6678 DSP,FPGA最大可支持4GByte容量DDR3 SDRAM,DSP可最大支持2G..._6678dsp adc

时间序列模型调查_时间序列模型的优缺点-程序员宅基地

文章浏览阅读5k次,点赞2次,收藏16次。RNN 模型Recurrent Neural Network (回流神经网络,有的译做递归神经网络)时间序列模型最常用最强大的的工具就是递归神经网络(recurrent neural network, RNN)。相比与普通神经网络的各计算结果之间相互独立的特点,RNN的每一次隐含层的计算结果都与当前输入以及上一次的隐含层结果相关。通过这种方法,RNN的计算结果便具备了记忆之前几次结果..._时间序列模型的优缺点

linux 中 tar \ zip 解压错误后撤回_linux撤销解压命令-程序员宅基地

文章浏览阅读233次。【代码】linux 中 tar \ zip 解压错误后撤回。_linux撤销解压命令

如何在水经微图APP中扫码添加图源?-程序员宅基地

文章浏览阅读236次。微图APP目前暂无数据同步功能,也没有帐号登录功能,但标绘导出功能是完全免费开放的,如果你觉得用起来还不错,欢迎分享给你的朋友。微图APP下载安装完成之后,会默认显示吉林一号卫星影像,你可以通过点击左下角的地图主菜单切换地图数据源。在“图源管理”界面中,点击右上角按钮,可以扫码添加微图Web版生成的图源二维码,在“编辑图源”界面中,你可以对地图名称和地图URL等相关参数进行编辑。在地图主菜单中,点击“地图”小图标,可以进入“图源管理”界面。这里,再为你分享在微图APP中通过扫二维码添加图源的方法。

Linux脏牛提权漏洞复现(CVE-2016-5195)_cve-2016-5195复现-程序员宅基地

文章浏览阅读605次。脏牛漏洞是一个严重的漏洞,可以允许攻击者获得管理员权限。为了保护系统的安全,请及时更新内核并采取其他安全措施。_cve-2016-5195复现