一個(gè)對(duì)象(在學(xué)習(xí)設(shè)計(jì)模式之前,需要比較了解面向?qū)ο笏枷耄┲回?fù)責(zé)一個(gè)特定的任務(wù),需要的朋友可以參考下
單例模式(職責(zé)模式):
簡(jiǎn)單的說,一個(gè)對(duì)象(在學(xué)習(xí)設(shè)計(jì)模式之前,需要比較了解面向?qū)ο笏枷耄┲回?fù)責(zé)一個(gè)特定的任務(wù);
單例類:
1、構(gòu)造函數(shù)需要標(biāo)記為private(訪問控制:防止外部代碼使用new操作符創(chuàng)建對(duì)象),單例類不能在其他類中實(shí)例化,只能被其自身實(shí)例化;
2、擁有一個(gè)保存類的實(shí)例的靜態(tài)成員變量
3、擁有一個(gè)訪問這個(gè)實(shí)例的公共的靜態(tài)方法(常用getInstance()方法進(jìn)行實(shí)例化單例類,通過instanceof操作符可以檢測(cè)到類是否已經(jīng)被實(shí)例化)
另外,需要?jiǎng)?chuàng)建__clone()方法防止對(duì)象被復(fù)制(克隆)
為什么要使用PHP單例模式?
1、php的應(yīng)用主要在于數(shù)據(jù)庫應(yīng)用, 所以一個(gè)應(yīng)用中會(huì)存在大量的數(shù)據(jù)庫操作, 使用單例模式, 則可以避免大量的new 操作消耗的資源。
2、如果系統(tǒng)中需要有一個(gè)類來全局控制某些配置信息, 那么使用單例模式可以很方便的實(shí)現(xiàn). 這個(gè)可以參看ZF的FrontController部分。
3、在一次頁面請(qǐng)求中, 便于進(jìn)行調(diào)試, 因?yàn)樗械拇a(例如數(shù)據(jù)庫操作類db)都集中在一個(gè)類中, 我們可以在類中設(shè)置鉤子, 輸出日志,從而避免到處var_dump, echo。
代碼實(shí)現(xiàn):
<111?111php111
/1**
* 設(shè)計(jì)模式之單例模式
* $_instance必須聲明為靜態(tài)的私有變量
* 構(gòu)造函數(shù)和析構(gòu)函數(shù)必須聲明為私有,防止外部程序new
* 類從而失去單例模式的意義
* getInstance()方法必須設(shè)置為公有的,必須調(diào)用此方法
* 以返回實(shí)例的一個(gè)引用
* ::操作符只能訪問靜態(tài)變量和靜態(tài)函數(shù)
* new對(duì)象都會(huì)消耗內(nèi)存
* 使用場(chǎng)景:最常用的地方是數(shù)據(jù)庫連接。
* 使用單例模式生成一個(gè)對(duì)象后,
* 該對(duì)象可以被其它眾多對(duì)象所使用。
*/
class Danli {
//保存類實(shí)例的靜態(tài)成員變量
private static $_instance;
//private標(biāo)記的構(gòu)造方法
private function __construct(){
echo 'This is a Constructed method;';
}
//創(chuàng)建__clone方法防止對(duì)象被復(fù)制克隆
public function __clone(){
trigger_error('Clone is not allow!',E_USER_ERROR);
}
//單例方法,用于訪問實(shí)例的公共的靜態(tài)方法
public static function getInstance(){
if(!(self::$_instance instanceof self)){
self::$_instance = new self;
}
return self::$_instance;
}
public function test(){
echo '調(diào)用方法成功';
}
}
//用new實(shí)例化private標(biāo)記構(gòu)造函數(shù)的類會(huì)報(bào)錯(cuò)
//$danli = new Danli();
//正確方法,用雙冒號(hào)::操作符訪問靜態(tài)方法獲取實(shí)例
$danli = Danli::getInstance();
$danli->test();
//復(fù)制(克隆)對(duì)象將導(dǎo)致一個(gè)E_USER_ERROR
$danli_clone = clone $danli;