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


Posted in PHP onFebruary 06, 2017

本文实例讲述了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 相关文章推荐
WINDOWS 2000下使用ISAPI方式安装PHP
Sep 05 PHP
PHP 图像尺寸调整代码
May 26 PHP
PHP 冒泡排序 二分查找 顺序查找 二维数组排序算法函数的详解
Jun 25 PHP
php function用法如何递归及return和echo区别
Mar 07 PHP
php警告Creating default object from empty value 问题的解决方法
Apr 02 PHP
PHP使用range协议实现输出文件断点续传代码实例
Jul 04 PHP
php中$_POST与php://input的区别实例分析
Jan 07 PHP
php读取文件内容的方法汇总
Jan 24 PHP
PHP框架性能测试报告
May 08 PHP
thinkphp中多表查询中防止数据重复的sql语句(必看)
Sep 22 PHP
PHP 实现从数据库导出到.csv文件方法
Jul 06 PHP
Laravel中日期时间处理包Carbon的简单使用
Sep 21 PHP
PHP第三方登录―QQ登录实现方法
Feb 06 #PHP
php计算给定日期所在周的开始日期和结束日期示例
Feb 06 #PHP
PHP实现判断数组是一维、二维或几维的方法
Feb 06 #PHP
php中get_magic_quotes_gpc()函数说明
Feb 06 #PHP
PHP数字前补0的自带函数sprintf 和number_format的用法(详解)
Feb 06 #PHP
浅谈php中fopen不能创建中文文件名文件的问题
Feb 06 #PHP
PHP 类与构造函数解析
Feb 06 #PHP
You might like
说说PHP的autoLoad自动加载机制
2012/09/27 PHP
PHP 下载文件时自动添加bom头的方法实例
2014/01/10 PHP
PHP获取链表中倒数第K个节点的方法
2018/01/18 PHP
PHP实现笛卡尔积算法的实例讲解
2019/12/22 PHP
AJAX架构之Dojo篇
2007/04/10 Javascript
关于javascript中的parseInt使用技巧
2009/09/03 Javascript
JavaScript 比较时间大小的代码
2010/04/24 Javascript
基于jQuery替换table中的内容并显示进度条的代码
2011/08/02 Javascript
javascript淡入淡出效果的实现思路
2012/03/31 Javascript
快速使用Bootstrap搭建传送带
2016/05/06 Javascript
JavaScript实现拖拽元素对齐到网格(每次移动固定距离)
2016/11/30 Javascript
Angular.js实现注册系统的实例详解
2016/12/18 Javascript
关于foreach循环中遇到的问题小结
2017/05/08 Javascript
jquery动态赋值id与动态取id方法示例
2017/08/21 jQuery
微信小程序之GET请求的实例详解
2017/09/29 Javascript
使用cropper.js裁剪头像的实例代码
2017/09/29 Javascript
从零开始学习搭建React脚手架项目
2018/08/23 Javascript
Vue框架下引入ActiveX控件的问题解决
2019/03/25 Javascript
小程序组件之自定义顶部导航实例
2019/06/12 Javascript
使用JQuery自动完成插件Auto Complete详解
2019/06/18 jQuery
微信小程序vant弹窗组件的实现方式
2020/02/21 Javascript
如何利用JavaScript编写一个格斗小游戏
2021/01/06 Javascript
python判断字符串是否包含子字符串的方法
2015/03/24 Python
代码分析Python地图坐标转换
2018/02/08 Python
python将.ppm格式图片转换成.jpg格式文件的方法
2018/10/27 Python
Python浮点数四舍五入问题的分析与解决方法
2019/11/19 Python
从numpy数组中取出满足条件的元素示例
2019/11/26 Python
利用jupyter网页版本进行python函数查询方式
2020/04/14 Python
使用OpenCV实现道路车辆计数的使用方法
2020/07/15 Python
HTML5网页音乐播放器的示例代码
2017/11/09 HTML / CSS
HTML5对比HTML4的主要改变和改进总结
2016/05/27 HTML / CSS
Evisu官方网站:日本牛仔品牌,时尚街头设计风格
2016/12/30 全球购物
医学生自我鉴定范文
2013/11/08 职场文书
计算机专业求职信
2014/06/02 职场文书
2014年最新领导班子整改方案
2014/09/27 职场文书
2015试用期转正工作总结
2014/12/12 职场文书