几乎所有面向对象的程序中,总有一两个资源被创建出来,在程序应用中持续被共享使用。例如,这样的一个资源,在一个电子商务程序的数据库连接中使用:这个连接在应用程序启动时初始化,程序于是可以有效的执行;当程序结束时,这个连接最终被断开并销毁。如果是你写的代码,没必要在每时每刻创建一个数据库连接,这样非常低效。已经建立好的连接应该能被你的代码简单重复的使用。这个问题就是,基于以上要求你将如何进行这个数据库连接?(或者连接其它被循环使用的唯一资源,比如一个开放文件或者一个队列。)
问题
你怎样确保一个特殊类的实例是独一无二的(它是这个类的唯一实例),并且它很存取容易呢?
解决方案
当然,全局变量是显而易见的解决方案。但它就像潘多拉的盒子(正确的判断来自经验,而错误的判断产生经验。这句谚语就是这个意思。),你的任何代码都能修改全局变量,这将不可避免的引起更多调试的意外。换句话说,全局变量的状态总是会出现一些问题的,(这里有一个关于全局变量使用问题不错的描述,http://c2.com/cgi/wiki?GlobalVariablesAreBad)。
当你需要一个特殊类的唯一实例时,使用这个名字叫单件的模式。基于单件模式的类能实例化和初始化这个类的一个实例,并且提供每时每刻绝对相同的连接。一般情况下使用名为getInstance()的静态方法实现。
单件模式是我们在开发中经常用到的一种设计模式,利用PHP5面向对象的特性,我们可以很容易的构建单件模式的应用,下面是单件模式在PHP中的几种实现方法:
static $instance = NULL ;
static function getInstance(){
if (self :: $instance == NULL ){
self :: $instance = new Stat ();
}
return self :: $instance ;
}
private function __construct(){
}
private function __clone(){
}
function sayHi(){
return " The Class is saying hi to u " ;
}
}
echo Stat :: getInstance() -> sayHi();
这是一种最通常的方式,在一个getInstance方法中返回唯一的类实例。
对这里例子稍加修改,便可以产生一个通用的方法,只要叫道任何你想用到单件的类里,就可以了。
function sayHi(){
return " The teacher smiling and said 'Hello ' " ;
}
static function getInstance(){
static $instance ;
if ( ! isset ( $instance )){
$c = __CLASS__ ;
$instance = new $c ;
}
return $instance ;
}
}
echo Teacher :: getInstance() -> sayHi();
最后一种是提供一个singleton类,然后通过调用getInstance方法,可以为任何一个类生产出一个实例来。
function getInstance( $class ){
static $instances = array ();
if ( ! array_key_exists ( $class , $instances )){
$instances [ $class ] = & new $class ;
}
$instance = $instances [ $class ];
return $instance ;
}
}
class People{
function sayHi(){
return ' Hello i am a people? ' ;
}
}
echo " <br /> " ;
echo singleton :: getInstance( ' People ' ) -> sayHi();
通过这三种方法,我们可以很容易的应用单件模式,如果能够结合工厂模式,将使我们的编程变得更有条理和效率。