今天来说一说单例模式。
由于我以前是做java开发的,在使用单例模式的时候,首先想到的想用饿汉式,然后发现在PHP中,有这样一个特性:因为PHP不支持在类定义时给类的成员变量赋予非基本类型的值。如表达式,new操作等等。所以了饿汉式这个就不行了。转而想要确保这个单例模式的原子性,发现PHP中也没有像JAVA中的线程安全问题。嘿嘿,你说PHP好不好?那么OK接下来就试试PHP的懒汉式单例模式了。
先不说,我先上我第一个版本的单例模式代码:
// 定义私有静态变量.此种方式为:懒汉式单例(PHP中只有这种方式) private static $instance = null; // 私有化构成方法 private function __construct(){ } // 提供获取实例的公共方法 public static function getInstance(){ if(!(self::$instance instanceof self)){ self::$instance = new self(); } return self::$instance; } // 私有__clone方法,禁止复制对象 private function __clone(){ }OK,这段代码看起很完美了,有注释,有格式的,没什么问题了吧。但是当我在使用的过程中,我发现了一下问题
我的A类是单例模式的,然后我的B类继承自A类,然后我调用如下方法:
$a = A::getInstance(); $b = B::getInstance(); var_dump($a === $b);输出的结果是:bool(true)
这个输出结果是什么意思呢?也就是说:B继承自A后,我本意是B也变成单例模式,那么A、B只是继承管理,他们的对象不应该相等,而现在两个的对象完全一样了,只能说明:通过
$b = B::getInstance();得到的对象,还是是A类的对象,那这是怎么回事?
问题出在self上,self的引用是在类被定义时就决定的,也就是说,继承了B的A,他的self引用仍然指向A。为了解决这个问题,在PHP 5.3中引入了后期静态绑定的特性。简单说是通过static关键字来访问静态的方法或者变量,与self不同,static的引用是由运行时决定。于是简单改写一下我们的代码,让单例模式可以复用。
class C { protected static $_instance = null; protected function __construct(){ } protected function __clone(){ } public function getInstance(){ if (static::$_instance === null) { static::$_instance = new static; } return static::$_instance; } } class D extends C{ protected static $_instance = null; } $c = C::getInstance(); $d = D::getInstance(); var_dump($c === $d);这是时候的输出就会变成:bool(false)
然后就可以达到,只要继承这个单例模式,那么它的子类也是单例模式。就可以达到完美复用的作用,不用每次需要单例模式都去写那么多重复代码了。注意上面的方法只有在PHP 5.3中才能使用,对于之前版本的PHP,还是老老实实为每个单例类写一个getInstance()方法吧。