php专区

 首页 > php专区 > PHP应用 > 常用功能 > PHP单例模式学习笔记详解 - php高级应用

PHP单例模式学习笔记详解 - php高级应用

分享到:
【字体:
导读:
          单例模式是php中一个为了简化大家开发及重复调用的一个功能,下面我来给各位朋友详细介绍单例模式用法。1 单例模式的概念顾名思义,单例模式...

PHP单例模式学习笔记详解

单例模式是php中一个为了简化大家开发及重复调用的一个功能,下面我来给各位朋友详细介绍单例模式用法。

1.单例模式的概念

顾名思义,单例模式只有一个实例,而且自行实例化,向全局提供这个实例,需要强调的是,单例模式,确保某个类只能有一个实例!

2.单例模式的三个要点

(1)需要一个静态变量来保存类的唯一实例,代码如下:

private static $_instance;

(2)构造函数和克隆函数必须为私有的,防止用户创建对象和复制实例对象,代码如下:

  1. private function __construct() 
  2. //私有化默认构造方法,防止外界直接实例化 
  3. private function __clone() 
  4. //私有化克隆方法,防止用户复制实例 

(3)必须提供一个公共的静态方法(一般为getInstance),从而返回一个唯一实例的引用,实例代码如下:

  1. public static function getInstance()     
  2. {     
  3.     if(! (self::$_instance instanceof self) )    
  4.     {     
  5.         self::$_instance = new self();     
  6.     }   
  7.     return self::$_instance;     
  8.    

3.php中使用单例模式的原因

PHP语言是一种解释型的脚本语言,这种运行机制使得每个PHP页面被解释执行后,所有的相关资源都会被回收,也就是说,PHP在语言级别上没有办法让某个对象常驻内存,这和asp.net、Java等编译型是不同的,比如在Java中单例会一直存在于整个应用程序的生命周期里,变量是跨页面级的,真正可以做到这个实例在应用程序生命周期中的唯一性。然而在PHP中,所有的变量无论是全局变量还是类的静态成员,都是页面级的,每次页面被执行时,都会重新建立新的对象,都会在页面执行完毕后被清空,这样似乎PHP单例模式就没有什么意义了,所以PHP单例模式我觉得只是针对单次页面级请求时出现多个应用场景并需要共享同一对象资源时是非常有意义的。

4.如何实现单例模式,代码如下:

  1. /** 
  2.  * 单例模式示例:Demo 
  3.  */ 
  4. class Demo{ 
  5.  //静态成员变量,用来保存全局实例 
  6.  private static $_instance
  7.  //私有化构造方法,保证外界不能直接实例化 
  8.  private function __construct(){ 
  9.  } 
  10.  //私有化克隆方法,防止用户复制实例 
  11.  private function __clone(){ 
  12.  } 
  13.  //返还此类的唯一实例 
  14.  public function getInstance(){ 
  15.   if(!(self::$_instance instanceof self)) 
  16.   { 
  17.    self::$_instance = new self(); 
  18.   } 
  19.   return self::$_instance
  20.  } 
  21.  //这是第一个测试方法 
  22.  public function test1Function(){ 
  23.   echo '这是第一个测试方法'
  24.  } 
  25.  //这是第二个测试方法 
  26.  public function test2Function(){ 
  27.   echo '这是第二个测试方法'
  28.  } 
  29. //正确的使用方法 
  30. @$demo = Demo::getInstance(); 
  31. $demo->test1Function(); 
  32. $demo->test2Function(); 
  33. //这样实例化会出错,因为构造方法为private 
  34. //$demo_new = new Demo; 
  35. //复制demo会出错,因为默认的clone方法为private 
  36. // $demo_clone = clone $demo; 
  37. ?> 

5.单利模式的应用场合

(1)应用与数据库的交互,多用于数据库的连接

(2)如果系统中需要一个类来全局控制配置信息,用单例模式可以很方便的实现

1、普通的数据库访问例子,代码如下:

  1. ...... 
  2. //初始化一个数据库句柄 
  3. $db = new DB(...); 
  4. //添加用户信息 
  5. $db->addUserInfo(...); 
  6. ...... 
  7. //在函数中访问数据库,查找用户信息 
  8. function getUserInfo() 
  9.     $db = new DB(...);//再次new 数据库类,和数据库建立连接 
  10.     $db = query(....);//根据查询语句访问数据库 
  11. ?> 

2、应用单例模式对数据库进行操作,实例代码如下:

  1. class DB   
  2. {   
  3.     private $_db;   
  4.     private static $_instance;   
  5.    
  6.     private function __construct(...)   
  7.     {   
  8.         $this->_db = pg_connect(...);//postgrsql   
  9.     }   
  10.    
  11.     private function __clone() {};  //覆盖__clone()方法,禁止克隆   
  12.    
  13.     public static function getInstance()   
  14.     {   
  15.         if(! (self::$_instance instanceof self) ) {   
  16.             self::$_instance = new self();   
  17.         }   
  18.         return self::$_instance;   
  19.     }   
  20.     public function addUserInfo(...) 
  21.     { 
  22.    
  23.     } 
  24.      public function getUserInfo(...) 
  25.     { 
  26.     } 
  27. //test 
  28. $db = DB::getInstance(); 
  29. $db->addUserInfo(...); 
  30. $db->getUserInfo(...); 
  31.  
  32. ?> 

深入理解,实例代码如下:

  1. class db { 
  2.  public $conn
  3.  public static $sql
  4.  public static $instance=null; 
  5.  private function __construct(){ 
  6.   require_once('db.config.php'); 
  7.   $this->conn = mysql_connect($db['host'],$db['user'],$db['password']); 
  8.   if(!mysql_select_db($db['database'],$this->conn)){ 
  9.    echo "失败"
  10.   }; 
  11.   mysql_query('set names utf8',$this->conn);   
  12.  } 
  13.  public static function getInstance(){ 
  14.   if(is_null(self::$instance)){ 
  15.    self::$instance = new db; 
  16.   } 
  17.   return self::$instance
  18.  } 
  19.  /** 
  20.   * 查询数据库 
  21.   */ 
  22.  public function select($table,$condition=array(),$field = array()){ 
  23.   $where=''
  24.   if(!emptyempty($condition)){ 
  25.     
  26.    foreach($condition as $k=>$v){ 
  27.     $where.=$k."='".$v."' and "
  28.    } 
  29.    $where='where '.$where .'1=1'
  30.   } 
  31.   $fieldstr = ''
  32.   if(!emptyempty($field)){ 
  33.     
  34.    foreach($field as $k=>$v){ 
  35.     $fieldstr.= $v.','
  36.    } 
  37.     $fieldstr = rtrim($fieldstr,','); 
  38.   }else
  39.    $fieldstr = '*'
  40.   } 
  41.   self::$sql = "select {$fieldstr} from {$table} {$where}"
  42.   $result=mysql_query(self::$sql,$this->conn); 
  43.   $resuleRow = array(); 
  44.   $i = 0; 
  45.   while($row=mysql_fetch_assoc($result)){ 
  46.    foreach($row as $k=>$v){ 
  47.     $resuleRow[$i][$k] = $v
  48.    } 
  49.    $i++; 
  50.   } 
  51.   return $resuleRow
  52.  } 
  53.  /** 
  54.   * 添加一条记录 
  55.   */ 
  56.   public function insert($table,$data){ 
  57.    $values = ''
  58.    $datas = ''
  59.    foreach($data as $k=>$v){ 
  60.     $values.=$k.','
  61.     $datas.="'$v'".','
  62.    } 
  63.    $values = rtrim($values,','); 
  64.    $datas   = rtrim($datas,','); 
  65.    self::$sql = "INSERT INTO  {$table} ({$values}) VALUES ({$datas})"
  66.   if(mysql_query(self::$sql)){ 
  67.    return mysql_insert_id(); 
  68.   }else
  69.    return false; 
  70.   }; 
  71.   } 
  72.   /** 
  73.    * 修改一条记录 
  74.    */ 
  75.  public function update($table,$data,$condition=array()){ 
  76.   $where=''
  77.   if(!emptyempty($condition)){ 
  78.     
  79.    foreach($condition as $k=>$v){ 
  80.     $where.=$k."='".$v."' and "
  81.    } 
  82.    $where='where '.$where .'1=1'
  83.   } 
  84.   $updatastr = ''
  85.   if(!emptyempty($data)){ 
  86.    foreach($data as $k=>$v){ 
  87.     $updatastr.= $k."='".$v."',"
  88.    } 
  89.    $updatastr = 'set '.rtrim($updatastr,','); 
  90.   } 
  91.   self::$sql = "update {$table} {$updatastr} {$where}"
  92.   return mysql_query(self::$sql); 
  93.  } 
  94.  /** 
  95.   * 删除记录 
  96.   */ 
  97.   public function delete($table,$condition){ 
  98.    $where=''
  99.   if(!emptyempty($condition)){ 
  100.     
  101.    foreach($condition as $k=>$v){ 
  102.     $where.=$k."='".$v."' and "
  103.    } 
  104.    $where='where '.$where .'1=1'
  105.   } 
  106.   self::$sql = "delete from {$table} {$where}"
  107.   return mysql_query(self::$sql); 
  108.    
  109.   } 
  110.  public static function getLastSql(){ 
  111.   echo self::$sql
  112.  } 
  113. $db = db::getInstance(); 
  114. //$list = $db->select('demo',array('name'=>'tom','password'=>'ds'),array 
  115. ('name','password')); 
  116. //echo $db->insert('demo',array('name'=>'最近你啦','password'=>'123')); 
  117. //echo $db->update('demo',array("name"=>'xxx',"password"=>'123'),array('id'=>1)); 
  118. echo $db->delete('demo',array('id'=>'2')); 
  119. db::getLastSql(); 
  120. echo "
    "
  121. ?> 

php中有很多的设计模式,其中的单例模式是我们写代码的时候较为常用的一种模式,它不但能够有效的减少new操作的资源消耗.而且能够很方便的对某些全局配置信息进行控制!希望大家在php学习中深刻理解单例模式的应用。

分享到:
从后台到webshell的一点思路 - php高级应...
从后台到webshell的一点思路 主要是利用后台对access数据库的“备份数据库”或“恢复数据库”功能,“备份的数据库路径”等变量没有过滤导致可以把任意文件后缀改为asp,从而得到webshell,msssql版的程序就直接应用了access版的代码,导致sql版照样可以利用,具体方法和分析可以参考angel的文章...
php memcache和memcached的区别 - php高...
php memcache和memcached的区别 首先要肯定的是,必须开启memcached服务,才能用memcache和memcached的php api,那么这两个针对memcached服务的api之间有什么区别呢? 1.memcache是PHP自己的扩展,而memcached是基于libmemcached的扩展。 2.memcache只提供一些基本功能,比如set、get、replace...
  •         php迷,一个php技术的分享社区,专属您自己的技术摘抄本、收藏夹。
  • 在这里……