PHP 單例模式理解

bykiss發表於2019-04-10
/**
 * 單例模式
 * @author YangYang <1812271619@qq.com>
 * 可以想成在一次http請求中只產生該類的一個物件(即只new classname一次)
 * 經典的例子是資料庫連線(redis,mongodb,memcache等)
 * 在一次http請求中我們可能需要對資料庫做增刪改查多條sql操作
 * 但是如果一次http請求中每執行一條sql我們就mysql_connect(),很明顯會導致伺服器資源的浪費
 * 為了節約資源,就可以通過單例模式來實現一次http請求只做一次mysql_connect()
 * 即將mysql_connect()放在類方法的__construct中,並將__construct方法做成私有,
 * 這樣只能通過getInstance()方法來獲得mysql_connect()的資源連線符
 * getInstance()方法中判斷是否已經存在myql連線符,如果存在就直接返回該連線符
 * 否則new classname()即呼叫了__construct方法執行了mysql_connect()得到了資源連線符,並返回連線符
 * 因為現在PHP已不再建議直接使用mysql函式進行資料庫操作,而是建議通過PDO進行資料庫操作,所以這裡寫一個簡易PDO連線的單例模式
 * 這裡只是講解單例原理,資料庫的防sql注入等問題不做考慮
 * 準備工作 資料庫:test  資料表:user  欄位:id name  記錄:1  CodeAnti
 * 最終執行結果: 資料表user中id=1這條記錄被刪除
 */
class SinglePDO
{
        private static $_instance = null;
        private $_pdo;

        //私有,防止外部直接例項化new SinglePDO(...)
        private function __construct($dsn,$dbUser,$dbPassword)
        {
                try{
                        $this->_pdo = new PDO($dsn,$dbUser,$dbPassword);
                        $this->_pdo->exec('set names utf8');
                }catch(PDOException $e){
                        die("Error:{$e->getMessage()}");
                }
        }

        //私有,防止克隆
        private function __clone(){}

        //獲取連線例項
        public static function getInstance($dsn,$dbUser,$dbPassword)
        {
                if(self::$_instance === null)
                        self::$_instance = new self($dsn,$dbUser,$dbPassword);

                return self::$_instance;
        }

        //執行sql
        public function execSql($sql)
        {
                $result = $this->_pdo->exec($sql);
                return $result;
        }
}

$dsn = "mysql:host=localhost;dbname=test";
$dbUser = "root";
$dbPassword = "";
$sql = "delete from user where id = 1";
$pdo = SinglePDO::getInstance($dsn,$dbUser,$dbPassword);
$result = $pdo->execSql($sql); //$pdo->execSql($sql)多次呼叫,但仍然是同一個pdo物件
print_r($result);
本作品採用《CC 協議》,轉載必須註明作者和本文連結

相關文章