首页 > 编程技术 > php

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

发布时间:2017-7-6 23:49

小编分享的这篇文章介绍了PHP单例模式定义与使用实例详解,非常实用,有兴趣的同学可以参考一下本文。

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

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

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

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

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

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

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

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

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

一个简单的例子:

 代码如下复制代码

classDB{

   private$_link;

   //   保持单例类的静态变量

   static$_instance;

   //   私有的构造函数

   privatefunction__construct(){

       $this->_link = @mysqli_connect(__HOST__, __USER__, __PASSWORD__, __DATABASE__);

       if(! ($this->_link)){

          echo'Something wrong occurs on the database connection!' 

       }

   }

   //   防止单例类被克隆

   privatefunction__clone(){}

   //   外界访问单例类实例的接口

   publicstaticfunctiongetInstance(){

       if(! (self::$_instanceinstanceofself)){

          self::$_instance=newself();

       }

       returnself::$_instance;

   }

}

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

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

 代码如下复制代码

classDB {

   /**

    * 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

    */

   privatefunction__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

    */

   privatefunction__clone(){}

   /**

    * for other object to get the instance of db

    * @return self::instance

    * @access public

    */

   publicstaticfunctiongetInstance(){

       if(! (self::$_instanceinstanceofself)){

          self::$_instance=newself();

       }

       returnself::$_instance;

   }

   /**

    * query

    * @param  sql string

    * @param  message string

    * @return   resource

    * @access public

    */

   publicfunctionquery($sql,$message){

       $result= @mysqli_query($this->$_link,$sql)ordie($message. mysqli_error($this->$_link));

       return$result;

   }

   /**

    * mysqli_num_rows

    * @param  result resource

    * @return   int

    * @access public

    */

   publicfunctionnum($result){

       return@mysqli_num_rows($result);

   }

   /**

    * mysqli_fetch_array

    * @param  result resource

    * @return   array

    * @access public

    */

   publicfunctionfetchArr($result){

       return@mysqli_fetch_array($result);

   }

   /**

    * mysqli_insert_id

    * @return   int

    * @access public

    */

   publicfunctionlast_id(){

       return@mysqli_insert_id($this->_link);  

   }

   /**

    * close the database connection

    * @param  result resource

    * @return   null

    * @access public

    */

   publicfunctionclose(){

       @mysqli_close($this->_link);

   }

   /**

    * fetch once result from the specific sql query

    * @param  sql string

    * @param  message string

    * @return   array

    * @access public

    */

   publicfunctionfetchArrOnce($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

    */

   publicfunctionfetchArrMore($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

    */

   publicfunctionfetchNum($sql,$message){

       $result=$this->query($sql,$message);

       $resultNum=$this->num($result);

       return$resultNum;

   }

   /**

    * mysqli_prepare

    * @param  sql string

    * @return   stmt object

    * @access public

    */

   publicfunctionprepare($sql){

       return@mysqli_prepare($this->_link,$sql);

   }

   /**

    * mysqli_stmt_execute

    * @param  stmt object

    * @param  message string

    * @return   bool

    * @access public

    */

   publicfunctionstmt_execute($stmt,$message){

       @mysqli_stmt_execute($stmt)ordie($message. mysqli_error($this->_link));

   }

}

使用:

 代码如下复制代码

define("__HOST__","localhost");

define("__USER__","root");

define("__PASSWORD__","");

define("__DATABASE__","eee");

$db= DB::getInstance();

小编推荐的这篇文章详细介绍了PHP第三方登录—QQ登录实现方法,非常实用,有兴趣的同学快来看看吧!

oAuth基本原理

接入QQ登录前置条件

申请AppID 和Appkey

登录QQ互联申请网站应用或移动应用接入

按照步骤申请成功后,创建应用即可看到对应的AppId和AppKey

引入官方SDK

添加测试回调地址

1、在本地添加一个虚拟主机**,比如域名为test.gz06.cn,然后在hosts文件中加入此域名

127.0.0.1       localhost test.gz06.cn

2、在QQ互联的应用信息编辑中将本地添加的测试回调域名加入到回调地址中,用 ; 好分隔,修改的时候要再次点击验证,然后保存即可

http://gz06.cn;http://test.gz06.cn/callback.php

3、引入官方SDK

4、SDK参数配置

访问下载到根目录下的SDK


http://test.gz06.cn/Connect2.1/

提示需要配置参数,点击进去

配置所需参数

小编推荐的这篇文章介绍了php计算给定日期所在周的开始日期和结束日期示例,非常实用,有兴趣的同学快来看看吧。
 代码如下 复制代码

<?php

/**

 * 取得给定日期所在周的开始日期和结束日期

 * @param string $gdate 日期,默认为当天,格式:YYYY-MM-DD

 * @param int $weekStart 一周以星期一还是星期天开始,0为星期天,1为星期一

 * @return array 数组array( "开始日期 ",  "结束日期");

 */

functiongetAWeekTimeSlot($gdate='',$weekStart= 0) {

 if(!$gdate){

 $gdate=date("Y-m-d");

 }

 $w=date("w",strtotime($gdate) );//取得一周的第几天,星期天开始0-6

 $dn=$w?$w-$weekStart: 6;//要减去的天数

 $st=date("Y-m-d",strtotime("$gdate  - ".$dn."  days ") );

 $en=date("Y-m-d",strtotime("$st  +6  days ") );

 returnarray($st,$en);//返回开始和结束日期

}

$timeSlot=getAWeekTimeSlot('2017-01-24',1);

echo"Week Start:{$timeSlot[0]}--->Week End: {$timeSlot[1]} ";

?>

运行结果如下:

Week Start:2017-01-23--->Week End: 2017-01-29

php中fopen不能创建中文文件名文件怎么办?本文详细介绍了浅谈php中fopen不能创建中文文件名文件的问题,遇到问题的同学可以试试文中方法解决一下。

之前网页的chartset用的是utf-8,文件也用utf-8,然后用fopen()创建一个中文文件名的文件时问题就出来了,文件名都是乱 码!

查看了很多文档试了不少方法都解决不了,本来想着用别的方法绕过这个问题,忽然脑子里闪过Windows默认的文字编码是ansi,然后再 baidu了一下,证实了这点,所以我的网页也应该是ansi编码才能使创建的文件名不会是乱码。

接着就着手验证,把网页都用ansi保存,去掉chartset语句,果然ok了,但是网页的内容就成乱码了,后来想起,这个网页还include 了别的网页,把include的网页也改成ansi保存,哈哈万事ok

编程这个工作真的很靠积累,如果我以前没看过Windows默认编码是ansi,那这个问题就不知何年何月才能解决了

ps:< meta content ="text/html; charset=utf-8" http -equiv ="Content-type" > 这个meta标记一定要放在<title></title>之前才有效的

后来又想到了一个更好的解决方法,网页还是用utf-8编码和保存,只是fopen()里的文件名参 数单独给它编下码就行,php有iconv() 这个改换编码的程序,把utf-8转成 gb2312就可以避免中文文件名为乱码了

test.htm

 代码如下 复制代码

<!DOCTYPE html PUBLIC "-//W3C//DTD HTML 4.01 Transitional//EN" "http://www.w3.org/TR/html4/loose.dtd">

<html>

<head>

<metacontent="text/html; charset=UTF-8"http-equiv="Content-Type">

<title>标题:{title}</title>

</head>

<body>

<b>此新闻的内容:</b>{content}

</body>

</html>

test.php

 代码如下 复制代码

<?php

 

  //实际应用中很可能是查询数据库取内容。

  $rows=array(array("替换标题1","替换内容1"),array("替换标题2","替换内容2"));

  $filename="tmp.htm";

  foreach($rowsas$id=>$val){

    $title=$val[0];

    $content=$val[1];

    $pagename="测试".$id.".html";

    //对文件名的编码,避免中文文件名乱码

    $pagename= iconv("UTF-8","GBK",$pagename);

     

    //读取模板

    $tmpfile=fopen($filename,"r");

    $string=fread($tmpfile,filesize($filename));

    $string=str_replace("{title}",$title,$string);

    $string=str_replace("{content}",$content,$string);

    fclose($tmpfile);

    //写新文件

    $newpage=fopen($pagename,"w");

    fwrite($newpage,$string);

    fclose($newpage);

     

  }

  echo"创建成功!";

?>

标签:[!--infotagslink--]

您可能感兴趣的文章: