PHP单例模式(精讲)

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

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

单例模式的要点有三个:web

一是某个类只能有一个实例;数据库

二是它必须自行建立这个实例;安全

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

<?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 操做消耗的资源。this

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

相关文章
相关标签/搜索