PHP單例模式應用示例【多次連接數(shù)據(jù)庫只實例化一次】
本文實例講述了PHP單例模式應用。分享給大家供大家參考,具體如下:
以前剛開始工作的時候經(jīng)常連接數(shù)據(jù)庫,每次用到數(shù)據(jù)庫的時候就要用new進行實例并連接一次,當時因為連接數(shù)據(jù)庫的次數(shù)不是很頻繁,所以也沒什么。后來主管對我說我現(xiàn)在這樣每次都連接數(shù)據(jù)庫的如果數(shù)據(jù)讀取頻繁的話對數(shù)據(jù)庫和系統(tǒng)造成的壓力會很大,讓我想想辦法能不能就連接一次數(shù)據(jù)庫然后再次用到的時候就不用new一個新的連接了,當時怎么也沒想到好的辦法,知道最近學到了單例模式才恍然大悟,當時主管是引導我用單例模式的,只怪我以前對開發(fā)模式不懂。好了廢話少說,下面來看單例模式:
單例模式(職責模式):
簡單的說,一個對象(在學習設計模式之前,需要比較了解面向?qū)ο笏枷耄┲回撠熞粋€特定的任務;
單例類:
1、構(gòu)造函數(shù)需要標記為private(訪問控制:防止外部代碼使用new操作符創(chuàng)建對象),單例類不能在其他類中實例化,只能被其自身實例化;
2、擁有一個保存類的實例的靜態(tài)成員變量
3、擁有一個訪問這個實例的公共的靜態(tài)方法(常用getInstance()方法進行實例化單例類,通過instanceof操作符可以檢測到類是否已經(jīng)被實例化)
另外,需要創(chuàng)建__clone()方法防止對象被復制(克隆)
為什么要使用PHP單例模式?
1、php的應用主要在于數(shù)據(jù)庫應用, 所以一個應用中會存在大量的數(shù)據(jù)庫操作, 使用單例模式, 則可以避免大量的new 操作消耗的資源。
2、如果系統(tǒng)中需要有一個類來全局控制某些配置信息, 那么使用單例模式可以很方便的實現(xiàn). 這個可以參看ZF的FrontController部分。
3、在一次頁面請求中, 便于進行調(diào)試, 因為所有的代碼(例如數(shù)據(jù)庫操作類db)都集中在一個類中, 我們可以在類中設置鉤子, 輸出日志,從而避免到處var_dump, echo。
代碼(官方文檔中的):
<?php
class Singletons{
// 保存類實例在此屬性中
private static $instance;
// 構(gòu)造方法聲明為private,防止直接創(chuàng)建對象
private function __construct()
{
echo 'Iam constructed';
}
// singleton 方法
public static function singleton()
{
if (!isset(self::$instance)) {
$c = __CLASS__;
self::$instance =new$c;
}
return self::$instance;
}
// Example類中的普通方法
public function bark()
{
echo 'Woof!';
}
// 阻止用戶復制對象實例
public function __clone()
{
trigger_error('Clone is not allowed.',E_USER_ERROR);
}
}//endclass singletons
//這個寫法會出錯,因為構(gòu)造方法被聲明為private
//$test= new Singletons();
// 下面將得到Example類的單例對象
$test= Singletons::singleton();
$test->bark();
// 復制對象將導致一個E_USER_ERROR.
$test_clone= clone $test;
?>
結(jié)果:
I am constructed! Woof!
Fatal error:
Clone is not allowed. in E:\APMServ5.2.6\www\htdocs\Lee\myprogram\other\class\singletons.phpon line 31
以下內(nèi)容源自網(wǎng)絡(可以參考學習一下):
單例模式的三個要點:
(1). 需要一個保存類的唯一實例的靜態(tài)成員變量:
private static $_instance;
(2). 構(gòu)造函數(shù)和克隆函數(shù)必須聲明為私有的,防止外部程序new類從而失去單例模式的意義:
private function __construct()
{
$this->_db = pg_connect('xxxx');
}
private function __clone()
{
}//覆蓋__clone()方法,禁止克隆
(3). 必須提供一個訪問這個實例的公共的靜態(tài)方法(通常為getInstance方法),從而返回唯一實例的一個引用
public static function getInstance()
{
if(! (self::$_instance instanceof self) )
{
self::$_instance = new self();
}
return self::$_instance;
}
二、為什么要使用單例模式?
1、PHP缺點:
PHP語言是一種解釋型的腳本語言,這種運行機制使得每個PHP頁面被解釋執(zhí)行后,所有的相關資源都會被回收。也就是說,PHP在語言級別上沒有辦法讓某個對象常駐內(nèi)存,這和asp.NET、Java等編譯型是不同的,比如在Java中單例會一直存在于整個應用程序的生命周期里,變量是跨頁面級的,真正可以做到這個實例在應用程序生命周期中的唯一性。然而在PHP中,所有的變量無論是全局變量還是類的靜態(tài)成員,都是頁面級的,每次頁面被執(zhí)行時,都會重新建立新的對象,都會在頁面執(zhí)行完畢后被清空,這樣似乎PHP單例模式就沒有什么意義了,所以PHP單例模式我覺得只是針對單次頁面級請求時出現(xiàn)多個應用場景并需要共享同一對象資源時是非常有意義的。
2、單例模式在PHP中的應用場合:
(1)、應用程序與數(shù)據(jù)庫交互
一個應用中會存在大量的數(shù)據(jù)庫操作,比如過數(shù)據(jù)庫句柄來連接數(shù)據(jù)庫這一行為,使用單例模式可以避免大量的new操作,因為每一次new操作都會消耗內(nèi)存資源和系統(tǒng)資源。
(2)、控制配置信息
如果系統(tǒng)中需要有一個類來全局控制某些配置信息,那么使用單例模式可以很方便的實現(xiàn).
三、如何實現(xiàn)單例模式?
1、普通的數(shù)據(jù)庫訪問例子:
<?php
......
//初始化一個數(shù)據(jù)庫句柄
$db = new DB(...);
//添加用戶信息
$db->addUserInfo(...);
......
//在函數(shù)中訪問數(shù)據(jù)庫,查找用戶信息
function getUserInfo()
{
$db = new DB(...);//再次new 數(shù)據(jù)庫類,和數(shù)據(jù)庫建立連接
$db = query(....);//根據(jù)查詢語句訪問數(shù)據(jù)庫
}
?>
2、應用單例模式對數(shù)據(jù)庫進行操作:
<?php
class DB
{
private $_db;
private static $_instance;
private function __construct(...)
{
$this->_db = pg_connect(...);//postgrsql
}
private function __clone() {}; //覆蓋__clone()方法,禁止克隆
public static function getInstance()
{
if(! (self::$_instance instanceof self) ) {
self::$_instance = new self();
}
return self::$_instance;
}
public function addUserInfo(...)
{
}
public function getUserInfo(...)
{
}
}
//test
$db = DB::getInstance();
$db->addUserInfo(...);
$db->getUserInfo(...);
?>
3、深入理解
<?php
class db {
public $conn;
public static $sql;
public static $instance=null;
private function __construct(){
require_once('db.config.php');
$this->conn = mysql_connect($db['host'],$db['user'],$db['password']);
if(!mysql_select_db($db['database'],$this->conn)){
echo "失敗";
};
mysql_query('set names utf8',$this->conn);
}
public static function getInstance(){
if(is_null(self::$instance)){
self::$instance = new db;
}
return self::$instance;
}
/**
* 查詢數(shù)據(jù)庫
*/
public function select($table,$condition=array(),$field = array()){
$where='';
if(!empty($condition)){
foreach($condition as $k=>$v){
$where.=$k."='".$v."' and ";
}
$where='where '.$where .'1=1';
}
$fieldstr = '';
if(!empty($field)){
foreach($field as $k=>$v){
$fieldstr.= $v.',';
}
$fieldstr = rtrim($fieldstr,',');
}else{
$fieldstr = '*';
}
self::$sql = "select {$fieldstr} from {$table} {$where}";
$result=mysql_query(self::$sql,$this->conn);
$resuleRow = array();
$i = 0;
while($row=mysql_fetch_assoc($result)){
foreach($row as $k=>$v){
$resuleRow[$i][$k] = $v;
}
$i++;
}
return $resuleRow;
}
/**
* 添加一條記錄
*/
public function insert($table,$data){
$values = '';
$datas = '';
foreach($data as $k=>$v){
$values.=$k.',';
$datas.="'$v'".',';
}
$values = rtrim($values,',');
$datas = rtrim($datas,',');
self::$sql = "INSERT INTO {$table} ({$values}) VALUES ({$datas})";
if(mysql_query(self::$sql)){
return mysql_insert_id();
}else{
return false;
};
}
/**
* 修改一條記錄
*/
public function update($table,$data,$condition=array()){
$where='';
if(!empty($condition)){
foreach($condition as $k=>$v){
$where.=$k."='".$v."' and ";
}
$where='where '.$where .'1=1';
}
$updatastr = '';
if(!empty($data)){
foreach($data as $k=>$v){
$updatastr.= $k."='".$v."',";
}
$updatastr = 'set '.rtrim($updatastr,',');
}
self::$sql = "update {$table} {$updatastr} {$where}";
return mysql_query(self::$sql);
}
/**
* 刪除記錄
*/
public function delete($table,$condition){
$where='';
if(!empty($condition)){
foreach($condition as $k=>$v){
$where.=$k."='".$v."' and ";
}
$where='where '.$where .'1=1';
}
self::$sql = "delete from {$table} {$where}";
return mysql_query(self::$sql);
}
public static function getLastSql(){
echo self::$sql;
}
}
$db = db::getInstance();
//$list = $db->select('demo',array('name'=>'tom','password'=>'ds'),array('name','password'));
//echo $db->insert('demo',array('name'=>'最近你啦','password'=>'123'));
//echo $db->update('demo',array("name"=>'xxx',"password"=>'123'),array('id'=>1));
echo $db->delete('demo',array('id'=>'2'));
db::getLastSql();
echo "<pre>";
?>
更多關于PHP相關內(nèi)容感興趣的讀者可查看本站專題:《php面向?qū)ο蟪绦蛟O計入門教程》、《PHP數(shù)組(Array)操作技巧大全》、《PHP基本語法入門教程》、《PHP運算與運算符用法總結(jié)》、《php字符串(string)用法總結(jié)》、《php+mysql數(shù)據(jù)庫操作入門教程》及《php常見數(shù)據(jù)庫操作技巧匯總》
希望本文所述對大家PHP程序設計有所幫助。
相關文章
php使用CutyCapt實現(xiàn)網(wǎng)頁截圖保存的方法
這篇文章主要介紹了php使用CutyCapt實現(xiàn)網(wǎng)頁截圖保存的方法,結(jié)合實例形式詳細分析了CutyCapt的下載、安裝及php使用CutyCapt進行截圖與保存的相關操作技巧,需要的朋友可以參考下2016-10-10
php實現(xiàn)將HTML頁面轉(zhuǎn)換成word并且保存的方法
這篇文章主要介紹了php實現(xiàn)將HTML頁面轉(zhuǎn)換成word并且保存的方法,結(jié)合實例形式分析了PHPWord工具的功能與使用方法,具有一定參考借鑒價值,需要的朋友可以參考下2016-10-10
使用PHP接收POST數(shù)據(jù),解析json數(shù)據(jù)
本篇文章是對使用PHP接收POST數(shù)據(jù)以及json數(shù)據(jù)進行了詳細的分析介紹,需要的朋友參考下2013-06-06
PHP實現(xiàn)JS中escape與unescape的方法
這篇文章主要介紹了PHP實現(xiàn)JS中escape與unescape的方法,通過json_encode和json_decode方法實現(xiàn)JS中escape與unescape函數(shù)的功能,需要的朋友可以參考下2016-07-07
docker?中搭建php環(huán)境經(jīng)驗分享
這篇文章主要介紹了docker?中搭建php環(huán)境經(jīng)驗分享的相關資料,需要的朋友可以參考下2023-09-09
深入講解PHP的對象注入(Object Injection)
這篇文章主要介紹了PHP中對象注入的相關資料,文中通過示例代碼介紹的非常詳細,相信對大家具有一定的參考價值,需要的朋友們下面來一起看看吧。2017-03-03

