php设计模式之单例模式

简介

首先明确一下单例模式:如果希望在系统中某个类的对象只能存在一个,那么单例模式是最好的解决方案。

接下来让我们提炼出单例模式的几个关键点:假如有一个单例类,叫做 Singleton,那么:

1 singletonObj 对象应该可以被系统中的任何对象使用

2 singletonObj 对象不应该被存储在会被覆写的全局变量中

3 系统中不应超过一个singletonObj对象,也就是说 A对象可以设置singletonObj对象的一个属性,

而B对象不需要通过其他任何对象就可以直接获得该属性的值。

 

为了解决这个问题,那么我们就从问题本身入手,也就是如何控制一个类被实例化,

让该类的实例化只能实例化一次,这个听起来似乎真的有些难,但是实际上很简单,只需要定义一个私有的构造方法即可:

 

代码



1 <?php
2 /**
3 * Created by PhpStorm.
4 * User: evolution
5 * Date: 14-12-7
6 * Time: 下午2:40
7 */
8 class Singleton {
9 //单一对象属性
10 private static $instance;
11 //定义一些全局变量需要存放属性
12 private $props = array();
13
14 //私有的构造方法
15 private function __construct(){
16 echo 'into construct!';
17 }
18
19 //返回单一实例
20 public static function getInstance () {
21 //判断是否已经有了实例化的对象
22 if(empty(self::$instance)) {
23 //可以被override (动态解析)
24 self::$instance = new static();
25 //不可以被override (静态解析)
26 //self::$instance = new self();
27 }
28
29 return self::$instance;
30 }
31
32 //设置属性
33 public function setProperty ( $key, $value) {
34 $this->props[$key] = $value;
35 }
36
37 //获取属性
38 public function getPeoperty ( $key ) {
39 return $this->props[$key];
40 }
41
42 }
43
44
45 $singleObj = Singleton::getInstance();
46 $singleObj->setProperty('root_path','/www');
47 $singleObj->setProperty('tmp_path','/tmp');
48
49 //接下来删除该单例对象,如果还能获取到刚刚添加的属性,说明使用的是同一个对象
50 unset($singleObj);
51
52 $singleObj = Singleton::getInstance();
53 echo $singleObj->getPeoperty('root_path');
54 echo $singleObj->getPeoperty('tmp_path');


 

实例

单例模式数据库处理类(只涉及到简单的原理)



1 /**
2 * 单例模式
3 */
4 class Db{
5 private static $mysqli;//该类中的唯一一个实例
6 private function __construct(){}//防止在外部实例化该类
7 private function __clone(){}//禁止通过复制的方式实例化该类
8 public static function connect(){//数据库连接方法
9 self::$mysqli = new MySQLi("localhost","root","","test");
10 self::$mysqli->query("set names utf8");
11 }
12 public static function select($table){//数据库操作方法,后续可以继续完善和添加......
13 self::connect();
14 $result = self::$mysqli->query("select * from ".$table);
15 $result_arr = array();
16 while($query = $result->fetch_assoc()){
17 $result_arr[] = $query;
18 }
19 return $result_arr;//结果集以数组的形式返回
20 }
21 }
22 $result = Db::select("article");
23 echo '<pre>';
24 print_r($result);
25 echo '</pre>';