單例模式的定義:保證一個類只有一個實例,並提供一個訪問它的全局訪問點。
PHP實現單例模式:
class Singleton { //創建靜態私有的變量保存該類對象 static private $instance; //防止使用new直接創建對象 private function __construct(){} //防止使用clone克隆對象 private function __clone(){} static public function getInstance() { //判斷$instance是否是Singleton的對象,不是則創建 if (!self::$instance instanceof self) { self::$instance = new self(); } return self::$instance; } public function test() { echo "我是一個單例模式"; } } $sing = Singleton::getInstance(); $sing->test(); $sing2 = new Singleton(); //Fatal error: Uncaught Error: Call to private Singleton::__construct() from invalid context in $sing3 = clone $sing; //Fatal error: Uncaught Error: Call to private Singleton::__clone() from context
PHP實現單例模式有什么意義:
沒有意義,PHP沒有線程概念,沒有異步代碼,不會常駐內存,所有的修改自己都可以控制,不需要保證類只有一個實例,有什么意義?你要在數據庫連接類中使用單例模式,萬一有的頁面需要連接兩個數據庫怎么辦?
真的沒有意義嗎?
當然不是。
看下代碼:
class A { public function show() { echo "我是A類函數,要調用B類函數和C類函數<br/>"; $b = new B(); $b->show(); $c = new C(); $c->show('A'); } } class B { public function show() { echo "我是B類函數,要調用C類函數<br/>"; $c = new C(); $c->show('B'); } } class C { public function show($parm) { echo "我是C類函數,現在被{$parm}類函數調用<br/>"; } } $a = new A(); $a->show();
這里調用的過程中,創建了兩個C類對象,我們知道,new是很耗資源的一種操作,如果這里C類是單例模式,只new一次就行了,這個想法是可以的,但我們一般用另外一種方法來做:
class A { public function show() { echo "我是A類函數,要調用B類函數和C類函數<br/>"; $b = Unit::getInstance('B'); $b->show(); $c = Unit::getInstance('C'); $c->show('A'); } } class B { public function show() { echo "我是B類函數,要調用C類函數<br/>"; $c = Unit::getInstance('C'); $c->show('B'); } } class C { public function __construct() { echo "C類對象被創建<br/>"; } public function show($parm) { echo "我是C類函數,現在被{$parm}類函數調用<br/>"; } } class Unit { static public function getInstance($class) { static $arr = null; if (!isset($arr[$class]) || !$arr[$class] instanceof $class) { $arr[$class] = new $class(); } return $arr[$class]; } } $a = new A(); $a->show();
執行代碼可以看到,C類只被實例化了一次,再實際操作中,我們一般使用靜態變量保存對象,變相實現單例模式。
單例模式和使用靜態變量實現單例的區別:
單例模式強制類只能創建一個對象,一般是安全或者統一界面展示的時候會用到,而靜態變量實現單例主要是為了減少new操作,並不能阻止new和clone操作。