PHP单例模式定义与使用实例详解

6年以前  |  阅读数:636 次  |  编程语言:PHP 

本文实例讲述了PHP单例模式定义与使用。分享给大家供大家参考,具体如下:

先简单的介绍一下单例模式。单例模式就是在应用程序中保持某一个类实例只存在一个,而且不可以受外部环境的影响而生成这个类的第二个实例。它的优点,实际点见,如果在WEB开发中,保持单一个数据操作类实例的存在,可以减少不必要的多余连接数据库资源的消耗,对于大型的软件开发来说,可以使用单例来维持程序的状态,使不同操作实现同步,因为单例一直占据内存,而从不会有副本。

而对于PHP,使用单例最常用的场合莫过于写一个数据库操作类。不过在PHP中实现单例,有以下规则:

1)单例类必须拥有一个现式声明的构造函数,并且是私有的。

2)单例类必须有一个静态变量来存储类的实例,这样可以保持这个单例类就只有那么一个实例。

3)单例类必须提供一个静态方法,供其他所有的对象应用这个单例。

为什么要满足以上三个条件呢:

1)因为单例类在整个应用程序运行时,只能被创造一次,而且这种创造是不是通过外部调用而完成,而是自身完成。所以单例类是自己实例化自己,所以其构造函数必须是私有。任何其他外部对象都不可以再次构造一个单例类的副本。

2)因为单例类只能够自己实例化自己,而又要为所有外部应用提供自己的实例,所以类内部必须有一个可供外界访问,而又是唯一不变的访问存储对象点,所以要提供一个静态变量去存储单例类自己实例化自己的那个实例对象。

3)因为单例类的构造函数是私有的,所以单例类必须提供一个外部接口供外部环境调用单例类,所以必须有一个静态方法,它可以初始化单例类或者返回单例类的对象的引用。

一个简单的例子:


    class DB{
       private $_link;
       //   保持单例类的静态变量
       static $_instance;
       //   私有的构造函数
       private function __construct(){
           $this->_link = @mysqli_connect(__HOST__, __USER__, __PASSWORD__, __DATABASE__);
           if(! ($this->_link)){
              echo 'Something wrong occurs on the database connection!';  
           }
       }
       //   防止单例类被克隆
       private function __clone(){}
       //   外界访问单例类实例的接口
       public static function getInstance(){
           if(! (self::$_instance instanceof self)){
              self::$_instance = new self();
           }
           return self::$_instance;
       }
    }

注意,以上定义的一个 __clone() 函数,防止单例类对象被克隆。

以下也是一个简单的数据库操作类的单例,供参考:


    class DB {
       /**
        * the database connection
        * @var   resource
        * @access private
        */
       private $_link;
       /**
        * the static instance of single db
        * @var   object
        * @access static
        */
       static $_instance;
       /**
        * construct the single object
        * @return null
        * @access private
        */
       private function __construct(){
           $this->_link = @mysqli_connect(__HOST__, __USER__, __PASSWORD__, __DATABASE__);
           if(! ($this->_link)){
              echo 'Something wrong occurs on the database connection!';  
           }
       }
       /**
        * empty clone
        * @return null
        * @access private
        */
       private function __clone(){}
       /**
        * for other object to get the instance of db
        * @return self::instance
        * @access public
        */
       public static function getInstance(){
           if(! (self::$_instance instanceof self)){
              self::$_instance = new self();
           }
           return self::$_instance;
       }
       /**
        * query
        * @param  sql string
        * @param  message string
        * @return   resource
        * @access public
        */
       public function query($sql,$message){
           $result = @mysqli_query($this->$_link, $sql) or die($message . mysqli_error($this->$_link));
           return $result;
       }
       /**
        * mysqli_num_rows
        * @param  result resource
        * @return   int
        * @access public
        */
       public function num($result){
           return @mysqli_num_rows($result);
       }
       /**
        * mysqli_fetch_array
        * @param  result resource
        * @return   array
        * @access public
        */
       public function fetchArr($result){
           return @mysqli_fetch_array($result);
       }
       /**
        * mysqli_insert_id
        * @return   int
        * @access public
        */
       public function last_id(){
           return @mysqli_insert_id($this->_link);   
       }
       /**
        * close the database connection
        * @param  result resource
        * @return   null
        * @access public
        */
       public function close(){
           @mysqli_close($this->_link);
       }
       /**
        * fetch once result from the specific sql query
        * @param  sql string
        * @param  message string
        * @return   array
        * @access public
        */
       public function fetchArrOnce($sql, $message){
           $result = $this->query($sql, $message);
           $row = $this->fetchArr($result);
           return $row;
       }
       /**
        * fetch all results from the specific sql query
        * @param  sql string
        * @param  message string
        * @return   array
        * @access public
        */
       public function fetchArrMore($sql, $message){
           $result = $this->query($sql, $message);
           $moreRow = array();
           while($row = $this->fetchArr($result)){
              $moreRow[] = $row;
           }
           return $moreRow;
       }
       /**
        * fetch the number of results from the specific sql query
        * @param  sql string
        * @param  message string
        * @return   array
        * @access public
        */
       public function fetchNum($sql, $message){
           $result = $this->query($sql, $message);
           $resultNum = $this->num($result);
           return $resultNum;
       }
       /**
        * mysqli_prepare
        * @param  sql string
        * @return   stmt object
        * @access public
        */
       public function prepare($sql){
           return @mysqli_prepare($this->_link, $sql);
       }
       /**
        * mysqli_stmt_execute
        * @param  stmt object
        * @param  message string
        * @return   bool
        * @access public
        */
       public function stmt_execute($stmt, $message){
           @mysqli_stmt_execute($stmt) or die($message . mysqli_error($this->_link));
       }
    }

使用:


    define("__HOST__", "localhost");
    define("__USER__", "root");
    define("__PASSWORD__", "");
    define("__DATABASE__", "eee");
    $db = DB::getInstance();

更多关于PHP相关内容感兴趣的读者可查看本站专题:《php面向对象程序设计入门教程》、《PHP基本语法入门教程》、《PHP运算与运算符用法总结》、《PHP网络编程技巧总结》、《PHP数组(Array)操作技巧大全》、《php字符串(string)用法总结》、《php+mysql数据库操作入门教程》及《php常见数据库操作技巧汇总

希望本文所述对大家PHP程序设计有所帮助。

 相关文章:
PHP分页显示制作详细讲解
SSH 登录失败:Host key verification failed
获取IMSI
将二进制数据转为16进制以便显示
文件下载
获取IMEI
贪吃蛇
双位运算符
发送邮件
PHP自定义函数获取搜索引擎来源关键字的方法
Java生成UUID
提取后缀名
年的日历图
在Zeus Web Server中安装PHP语言支持
让你成为最历害的git提交人
Yii2汉字转拼音类的实例代码
再谈PHP中单双引号的区别详解
指定应用ID以获取对应的应用名称
Python 2与Python 3版本和编码的对比
php封装的page分页类完整实例