php實現單例模式的方法及含義

來源:互聯網
上載者:User
這篇文章主要介紹了php實現單例模式的方法,告訴大家為什麼使用單例模式,感興趣的朋友可以參考一下

一、什麼是單例模式?1、含義 作為對象的建立模式,單例模式確保某一個類只有一個執行個體,而且自行執行個體化並向整個系統全域地提供這個執行個體。它不會建立執行個體副本,而是會向單例類內部儲存的執行個體返回一個引用。
2、單例模式的三個要點:(1). 需要一個儲存類的唯一執行個體的靜態成員變數:
private static $_instance;
(2). 建構函式和複製函數必須聲明為私人的,防止外部程式new類從而失去單例模式的意義:

private function __construct()  {    $this->_db = pg_connect('xxxx'); }  private function __clone() { }//覆蓋__clone()方法,禁止複製

(3). 必須提供一個訪問這個執行個體的公用的靜態方法(通常為getInstance方法),從而返回唯一執行個體的一個引用

public static function getInstance()  {    if(! (self::$_instance instanceof self) )    {      self::$_instance = new self();    }   return self::$_instance;   }

二、為什麼要使用單例模式?
1、PHP缺點:

PHP語言是一種解釋型的指令碼語言,這種運行機制使得每個PHP頁面被解釋執行後,所有的相關資源都會被回收。也就是說,PHP在語言層級上沒有辦法讓某個對象常駐記憶體,這和asp.net、Java等編譯型是不同的,比如在Java中單例會一直存在於整個應用程式的生命週期裡,變數是跨頁面級的,真正可以做到這個執行個體在應用程式生命週期中的唯一性。然而在PHP中,所有的變數無論是全域變數還是類的靜態成員,都是頁面級的,每次頁面被執行時,都會重建立立新的對象,都會在頁面執行完畢後被清空,這樣似乎PHP單例模式就沒有什麼意義了,所以PHP單例模式我覺得只是針對單次頁面級請求時出現多個應用情境並需要共用同一對象資源時是非常有意義的。

2、單例模式在PHP中的應用場合:(1)、應用程式與資料庫互動
一個應用中會存在大量的資料庫操作,比如過資料庫控制代碼來串連資料庫這一行為,使用單例模式可以避免大量的new操作,因為每一次new操作都會消耗記憶體資源和系統資源。
(2)、控制配置資訊
如果系統中需要有一個類來全域控制某些配置資訊, 那麼使用單例模式可以很方便的實現.

三、如何?單例模式?1、普通的資料庫訪問例子:

<?php ...... //初始化一個資料庫控制代碼 $db = new DB(...);  //添加使用者資訊 $db->addUserInfo(...);  ......  //在函數中訪問資料庫,尋找使用者資訊 function getUserInfo() {   $db = new DB(...);//再次new 資料庫類,和資料庫建立串連   $db = query(....);//根據查詢語句訪問資料庫 }  ?>

2、應用單例模式對資料庫進行操作:

<?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(...);   ?>

下面的代碼是PDO操作資料庫類的一個封裝,採用了單例模式:

<?php/** * MyPDO */class MyPDO{  protected static $_instance = null;  protected $dbName = '';  protected $dsn;  protected $dbh;    /**   * 構造   *    * @return MyPDO   */  private function __construct($dbHost, $dbUser, $dbPasswd, $dbName, $dbCharset)  {    try {      $this->dsn = 'mysql:host='.$dbHost.';dbname='.$dbName;      $this->dbh = new PDO($this->dsn, $dbUser, $dbPasswd);      $this->dbh->exec('SET character_set_connection='.$dbCharset.', character_set_results='.$dbCharset.', character_set_client=binary');    } catch (PDOException $e) {      $this->outputError($e->getMessage());    }  }    /**   * 防止複製   *    */  private function __clone() {}    /**   * Singleton instance   *    * @return Object   */  public static function getInstance($dbHost, $dbUser, $dbPasswd, $dbName, $dbCharset)  {    if (self::$_instance === null) {      self::$_instance = new self($dbHost, $dbUser, $dbPasswd, $dbName, $dbCharset);    }    return self::$_instance;  }    /**   * Query 查詢   *   * @param String $strSql SQL語句   * @param String $queryMode 查詢方式(All or Row)   * @param Boolean $debug   * @return Array   */  public function query($strSql, $queryMode = 'All', $debug = false)  {    if ($debug === true) $this->debug($strSql);    $recordset = $this->dbh->query($strSql);    $this->getPDOError();    if ($recordset) {      $recordset->setFetchMode(PDO::FETCH_ASSOC);      if ($queryMode == 'All') {        $result = $recordset->fetchAll();      } elseif ($queryMode == 'Row') {        $result = $recordset->fetch();      }    } else {      $result = null;    }    return $result;  }    /**   * Update 更新   *   * @param String $table 表名   * @param Array $arrayDataValue 欄位與值   * @param String $where 條件   * @param Boolean $debug   * @return Int   */  public function update($table, $arrayDataValue, $where = '', $debug = false)  {    $this->checkFields($table, $arrayDataValue);    if ($where) {      $strSql = '';      foreach ($arrayDataValue as $key => $value) {        $strSql .= ", `$key`='$value'";      }      $strSql = substr($strSql, 1);      $strSql = "UPDATE `$table` SET $strSql WHERE $where";    } else {      $strSql = "REPLACE INTO `$table` (`".implode('`,`', array_keys($arrayDataValue))."`) VALUES ('".implode("','", $arrayDataValue)."')";    }    if ($debug === true) $this->debug($strSql);    $result = $this->dbh->exec($strSql);    $this->getPDOError();    return $result;  }    /**   * Insert 插入   *   * @param String $table 表名   * @param Array $arrayDataValue 欄位與值   * @param Boolean $debug   * @return Int   */  public function insert($table, $arrayDataValue, $debug = false)  {    $this->checkFields($table, $arrayDataValue);    $strSql = "INSERT INTO `$table` (`".implode('`,`', array_keys($arrayDataValue))."`) VALUES ('".implode("','", $arrayDataValue)."')";    if ($debug === true) $this->debug($strSql);    $result = $this->dbh->exec($strSql);    $this->getPDOError();    return $result;  }    /**   * Replace 覆蓋方式插入   *   * @param String $table 表名   * @param Array $arrayDataValue 欄位與值   * @param Boolean $debug   * @return Int   */  public function replace($table, $arrayDataValue, $debug = false)  {    $this->checkFields($table, $arrayDataValue);    $strSql = "REPLACE INTO `$table`(`".implode('`,`', array_keys($arrayDataValue))."`) VALUES ('".implode("','", $arrayDataValue)."')";    if ($debug === true) $this->debug($strSql);    $result = $this->dbh->exec($strSql);    $this->getPDOError();    return $result;  }    /**   * Delete 刪除   *   * @param String $table 表名   * @param String $where 條件   * @param Boolean $debug   * @return Int   */  public function delete($table, $where = '', $debug = false)  {    if ($where == '') {      $this->outputError("'WHERE' is Null");    } else {      $strSql = "DELETE FROM `$table` WHERE $where";      if ($debug === true) $this->debug($strSql);      $result = $this->dbh->exec($strSql);      $this->getPDOError();      return $result;    }  }    /**   * execSql 執行SQL語句   *   * @param String $strSql   * @param Boolean $debug   * @return Int   */  public function execSql($strSql, $debug = false)  {    if ($debug === true) $this->debug($strSql);    $result = $this->dbh->exec($strSql);    $this->getPDOError();    return $result;  }    /**   * 擷取欄位最大值   *    * @param string $table 表名   * @param string $field_name 欄位名   * @param string $where 條件   */  public function getMaxValue($table, $field_name, $where = '', $debug = false)  {    $strSql = "SELECT MAX(".$field_name.") AS MAX_VALUE FROM $table";    if ($where != '') $strSql .= " WHERE $where";    if ($debug === true) $this->debug($strSql);    $arrTemp = $this->query($strSql, 'Row');    $maxValue = $arrTemp["MAX_VALUE"];    if ($maxValue == "" || $maxValue == null) {      $maxValue = 0;    }    return $maxValue;  }    /**   * 擷取指定列的數量   *    * @param string $table   * @param string $field_name   * @param string $where   * @param bool $debug   * @return int   */  public function getCount($table, $field_name, $where = '', $debug = false)  {    $strSql = "SELECT COUNT($field_name) AS NUM FROM $table";    if ($where != '') $strSql .= " WHERE $where";    if ($debug === true) $this->debug($strSql);    $arrTemp = $this->query($strSql, 'Row');    return $arrTemp['NUM'];  }    /**   * 擷取表引擎   *    * @param String $dbName 庫名   * @param String $tableName 表名   * @param Boolean $debug   * @return String   */  public function getTableEngine($dbName, $tableName)  {    $strSql = "SHOW TABLE STATUS FROM $dbName WHERE Name='".$tableName."'";    $arrayTableInfo = $this->query($strSql);    $this->getPDOError();    return $arrayTableInfo[0]['Engine'];  }    /**   * beginTransaction 事務開始   */  private function beginTransaction()  {    $this->dbh->beginTransaction();  }    /**   * commit 事務提交   */  private function commit()  {    $this->dbh->commit();  }    /**   * rollback 交易回復   */  private function rollback()  {    $this->dbh->rollback();  }    /**   * transaction 通過交易處理多條SQL語句   * 調用前需通過getTableEngine判斷表引擎是否支援事務   *   * @param array $arraySql   * @return Boolean   */  public function execTransaction($arraySql)  {    $retval = 1;    $this->beginTransaction();    foreach ($arraySql as $strSql) {      if ($this->execSql($strSql) == 0) $retval = 0;    }    if ($retval == 0) {      $this->rollback();      return false;    } else {      $this->commit();      return true;    }  }  /**   * checkFields 檢查指定欄位是否在指定資料表中存在   *   * @param String $table   * @param array $arrayField   */  private function checkFields($table, $arrayFields)  {    $fields = $this->getFields($table);    foreach ($arrayFields as $key => $value) {      if (!in_array($key, $fields)) {        $this->outputError("Unknown column `$key` in field list.");      }    }  }    /**   * getFields 擷取指定資料表中的全部欄位名   *   * @param String $table 表名   * @return array   */  private function getFields($table)  {    $fields = array();    $recordset = $this->dbh->query("SHOW COLUMNS FROM $table");    $this->getPDOError();    $recordset->setFetchMode(PDO::FETCH_ASSOC);    $result = $recordset->fetchAll();    foreach ($result as $rows) {      $fields[] = $rows['Field'];    }    return $fields;  }    /**   * getPDOError 捕獲PDO錯誤資訊   */  private function getPDOError()  {    if ($this->dbh->errorCode() != '00000') {      $arrayError = $this->dbh->errorInfo();      $this->outputError($arrayError[2]);    }  }    /**   * debug   *    * @param mixed $debuginfo   */  private function debug($debuginfo)  {    var_dump($debuginfo);    exit();  }    /**   * 輸出錯誤資訊   *    * @param String $strErrMsg   */  private function outputError($strErrMsg)  {    throw new Exception('MySQL Error: '.$strErrMsg);  }    /**   * destruct 關閉資料庫連接   */  public function destruct()  {    $this->dbh = null;  }}?>

調用方法:

<?phprequire 'MyPDO.class.php';$db = MyPDO::getInstance('localhost', 'root', '123456', 'test', 'utf8');$db->query("select count(*) frome table");$db->destruct();?>

總結:以上就是本篇文的全部內容,希望能對大家的學習有所協助。

聯繫我們

該頁面正文內容均來源於網絡整理,並不代表阿里雲官方的觀點,該頁面所提到的產品和服務也與阿里云無關,如果該頁面內容對您造成了困擾,歡迎寫郵件給我們,收到郵件我們將在5個工作日內處理。

如果您發現本社區中有涉嫌抄襲的內容,歡迎發送郵件至: info-contact@alibabacloud.com 進行舉報並提供相關證據,工作人員會在 5 個工作天內聯絡您,一經查實,本站將立刻刪除涉嫌侵權內容。

A Free Trial That Lets You Build Big!

Start building with 50+ products and up to 12 months usage for Elastic Compute Service

  • Sales Support

    1 on 1 presale consultation

  • After-Sales Support

    24/7 Technical Support 6 Free Tickets per Quarter Faster Response

  • Alibaba Cloud offers highly flexible support services tailored to meet your exact needs.