單例類:
1.構造函數需要標記為private,單例類不能再其他類中實例化,只能被其自身實例化
2.擁有一個保存類的實例靜態成員變量
3.擁有一個訪問這個實例的公共的靜態方法。[常用getInstance()方法進行實例化單例類,通過instanceof操作符可以檢測到此類是否已經被實例化]
注:需要創建__clone()方法防治對象被復制
作用:
1.php的應用主要用於數據庫,所以一個應用中會存在大量的數據庫操作,使用單例模式,則可以避免大量new操作消耗的資源
2.如果系統中需要一個類來全局控制某些配置信息,那麼使用單例模式可以很方便的實現。參照ZF的FrontController部分
3.在一個頁面請求匯總,便於進行調試,因為所有的代碼都集中在一個類中,我們可以在類中設置鉤子,輸出日志,從而避免到處var_dump,echo。
php單例模式的例子。
代碼如下 復制代碼
<?php
/**
* 單例模式
*/
class DanLi{
//靜態成員變量
private static $_instance;
//私有的構造方法
private function __construct(){
}
//防止對象被克隆
public function __clone(){
trigger_error('Clone is not allow!',E_USER_ERROR);
}
public static function getInstance(){
if(!(self::$_instance instanceof self)){
self::$_instance = new self;
}
return self::$_instance;
}
public function test(){
echo "ok";
}
}
//錯誤:$danli = new DanLi(); $danli_clone = clone $danli;
//正確:$danli = DanLi::getInstance(); $danli->test();
?>
下面我們討論下為什麼要使用PHP單例模式?
多數人都是從單例模式的字面上的意思來理解它的用途, 認為這是對系統資源的節省, 可以避免重復實例化, 是一種"計劃生育". 而PHP每次執行完頁面都是會從內存中清理掉所有的資源. 因而PHP中的單例實際每次運行都是需要重新實例化的, 這樣就失去了單例重復實例化的意義了. 單單從這個方面來說, PHP的單例的確有點讓各位失望. 但是單例僅僅只有這個功能和應用嗎? 答案是否定的,我們一起來看看。
1. php的應用主要在於數據庫應用, 所以一個應用中會存在大量的數據庫操作, 在使用面向對象的方式開發時(廢話), 如果使用單例模式, 則可以避免大量的new 操作消耗的資源。
2. 如果系統中需要有一個類來全局控制某些配置信息, 那麼使用單例模式可以很方便的實現. 這個可以參看zend Framework的FrontController部分。
3. 在一次頁面請求中, 便於進行調試, 因為所有的代碼(例如數據庫操作類db)都集中在一個類中, 我們可以在類中設置鉤子, 輸出日志,從而避免到處var_dump, echo。
代碼如下 復制代碼<?php
/**
* 設計模式之單例模式
* $_instance必須聲明為靜態的私有變量
* 構造函數和析構函數必須聲明為私有,防止外部程序new
* 類從而失去單例模式的意義
* getInstance()方法必須設置為公有的,必須調用此方法
* 以返回實例的一個引用
* ::操作符只能訪問靜態變量和靜態函數
* new對象都會消耗內存
* 使用場景:最常用的地方是數據庫連接。
* 使用單例模式生成一個對象後,
* 該對象可以被其它眾多對象所使用。
*/
class Example
{
//保存例實例在此屬性中
private static $_instance;
//構造函數聲明為private,防止直接創建對象
private function __construct()
{
echo 'I am Construceted';
}
//單例方法
public static function singleton()
{
if(!isset(self::$_instance))
{
$c=__CLASS__;
self::$_instance=new $c;
}
return self::$_instance;
}
//阻止用戶復制對象實例
public function __clone()
{
trigger_error('Clone is not allow' ,E_USER_ERROR);
}
function test()
{
echo("test");
}
}
// 這個寫法會出錯,因為構造方法被聲明為private
$test = new Example;
// 下面將得到Example類的單例對象
$test = Example::singleton();
$test->test();
// 復制對象將導致一個E_USER_ERROR.
$test_clone = clone $test;
?>