个人理解
意图:保证一个类仅有一个实例,并提供一个能够访问它的全局访问点。
主要解决:一个类的频繁的实例化、销毁,或者需要保证全局只有一个此类的实例化对象,另外,也可以防止随意实例化而占用过多的内存空间。
实现方式:将类的初始化函数私有化,使得外部不可直接实例化对象。然后在类的内部(由于初始化函数已经私有化了,所以必须在类的内部)设定一个能够获取到此类的实例化对象的静态方法,方法内部实现实例化类,并返回。
简单的单例模式
简单的单例模式非常简单,举个例子
<?php
//单例类
class People
{
private static $people;
//私有化初始化对象,也可以修改为protected
private function __construct()
{
}
//静态方法
public static function getPeople()
{
if (empty(self::$people)) {
self::$people = new People();
}
return self::$people;
}
}
这就是一个非常简单的单例模式,但是存在如下几个问题:
- 此类可以被继承,是否符合需求,因为继承后,此类还是会被实例化两次。
- 此类可以被克隆,还记得原型模式中的clone吗,如果被克隆了,也会违背我们需要保持唯一的初衷。
为了解决这个两个问题,还需要对简单的单例模式进行修改。
不可继承
<?php
//单例类(加入了final关键词,防止被继承)
final class People
{
private static $people;
//私有化初始化对象,也可以修改为protected
private function __construct()
{
}
//静态方法(因为People加了,所以方法的final可加可不加)
final public static function getPeople()
{
if (empty(self::$people)) {
self::$people = new People();
}
return self::$people;
}
}
$people = People::getPeople();
不可克隆
<?php
//单例类(加入了final关键词,防止被继承)
final class People
{
private static $people;
private $age = 10;
//私有化初始化对象,也可以修改为protected
private function __construct()
{
}
//静态方法(因为People加了,所以方法的final可加可不加)
final public static function getPeople()
{
if (empty(self::$people)) {
self::$people = new People();
}
return self::$people;
}
//加入了防止被克隆
public function __clone()
{
return self::$people;
}
public function setAge($age)
{
$this->age = $age;
}
public function showAge()
{
echo $this->age . '<hr/>';
}
}
$people = People::getPeople();
$people1 = clone $people;
$people->setAge(12);
$people->showAge();
$people1->showAge();
如果是这样实现的,则不会起作用。显示的结果为:
通过查看手册发现,__clone()方法的返回值并不是clone的结果值,这里我以前理解错了。只不过是会调用__clone()方法而已,所以如果需要禁用克隆,需要把__clone方法也私有化。
<?php
final class People
{
//上面的一样...
//加入了防止被克隆
protected function __clone(){}
//下面的一样...
}
这是如果再去clone的话,将会报错。
其他名词
在《大话设计模式》中出现了一对名词“饿汉式加载”与“懒汉式加载”。特征是,“饿汉式加载”是不需要静态方法的,只需要在静态变量中直接实例化对象,而“懒汉式加载”就是上面所说的单例模式。经过本人测试,PHP好像并不支持在变量上直接实例化对象,所以我就不贴代码了(如果有朋友发现了可以,可以提醒我一下,不胜感激)。
饿汉式加载代码(这应该是C#的代码吧):
参考资料
《大话设计模式》
菜鸟教程:http://www.runoob.com/design-pattern/singleton-pattern.html
下一篇
初识设计模式——桥接模式