在编写程序时经常会遇到一此典型的问题或需要完成某种特定需求,设计模式就是针对这些问题和需求,在大量的实践中总结和理论化之后优选的代码结构编程风格,以及解决问题的思考方式。
设计模式就像是经典的棋谱。不同的棋局,使用不同的棋谐,免得自己再去思考和模索。本节将针对PHP应用程序中最常用的两种设计模式进行详细讲解。
9.1 单例模式
单例模式是PHP中的一种设计模式,它是指在设计一个类时,需要保证在整个程序运行期间针对该类只存在一个实例对象。
就像世界上只有一个月亮,假设现在要设计一个类表示月亮,该类只能有一个实例对象,否则就违背了事实。
在讲解单例设计模式之前,通过一个案例来演示在什么情况时需要使用单例模式,如例9-23 所示
例9-23
1 <?php 2 class dbHelper{ 3 private $conn = null; 4 public function __construct(){ 5 //打开一个到 MySQL 服务器的连接 6 $this->conn = mysql_connect("localhost","root",""); 7 echo "得到一个conn<br/>"; 8 } 9 } 10 $db1 = new dbHelper(); 11 $db2 = new dbHelper(); 12 if($db1 === $db2){ 13 echo "一个对象<br/>"; 14 } else { 15 echo "两个对象<br/>"; 16 } 17 ?>
运行结果
得到一个conn
得到一个conn
两个对象
从上例中可以看出,实例化类dbHelper的两个对象请求的数据库连接是两个不同的连接,而在实际开发中,有时会有这样的需求,
在一次HTTP 请求中,保证某个类的对象实例只能有一个。这样可以节省资源开销,此时可以使用单例模式。
单例模式(Singleton)用于为一个类生成一个唯一的对象。(请记住名词 “”三私一公“”)
私有静态属性
私有构造方法
私有克隆方法
公有静态调用队象方法
将上面的dbHelper 类使用单例模式来实现,如例9-24所示
例9-24
<?php class dbHelper{ private static $instance = null;//定义一个私有的静态属性$instance //声明一个构造方法 private function __construct(){ $this->conn = mysql_connect("localhost","root",""); echo "得到一个conn<br/>"; } //只有通过这个方法才能返回本类的对象,该方法是静态方法 public static function getInstance(){ //如果本类中的$instance为空,说明它还没有被实例化过 if(self::$instance == null){ self::$instance = new self();//实例化本类对象 } return self::$instance;//返回本类的对象 } //阻止用户复制对象实例 public function __clone(){ } } $db1 = dbHelper::getInstance(); $db2 = dbHelper::getInstance(); if($db1 === $db2){ echo "同一个对象"; }else{ echo "不是同一个对象"; } ?>
运行结果
得到一个conn
同一个对象
在上例中,dbHelper类的构造方法使用了private 关键字进行了修饰,即不能在类定义之外使用new来创建对象。
如此一来就只能通过类 名直接调用getinstance0静态方法来创建对象。在第3行代码声明了一个私有的静态属性$instance.
将实例化的对象赋值给它,再判断该属性,如果已经有值,就直接返回,如果其值为null, 就先实例化对象,这样就能保证dbHelper类只能被实例化一次。
最后增加了一个私有的魔术方法_ clone0. 用于防止用户通过clone方法复制对象。
9.2 工厂模式
工厂模式的作用就是“生产”对象。工厂方法的参数是要生成对象的类名。
为了方便理解工厂模式的作用,接下来通过一个案例来演示如何使用工厂模式获取MySQL和sQLite的驱动对象。
首先在根目录下创建MySQLphp文件。示例代码如下:
1 <?php 2 class MySQL{ 3 4 //操作SQL的驱动类 5 6 }
然后在根目录下创建SQLite.php文件。示例代码如下:
1 <?php 2 class SQLite{ 3 4 //操作SQLite的驱动类 5 6 }
最后定义一个工厂方法来获取各驱动对象,代码如例9-25所示
例9-25
1 <?php 2 header('Content-Type: textml;charset=utf-8'); 3 class Db{ 4 //工厂方法 5 public static function factory($type){ 6 if (include_once $type . '.php') { 7 $classname = $type; 8 return new $classname(); 9 } else { 10 echo "出错了!"; 11 } 12 } 13 } 14 //获取MySQL驱动对象 15 $mysql = Db::factory('MySQL'); 16 //获取SQLite驱动对象 17 $sqlite = Db::factory('SQLite'); 18 var_dump($mysql); 19 var_dump($sqlite); 20 ?>
运行结果
object (MySQL) [1]
object (SQLite) [2]
上例中,第5行代码定义了一个静态方法factor(), 这就是工厂方法,该方法的参数为类名。
第6- 11行代码用于判断类名与参数是否相同,如果相同则创建该类的对象,否则输出“出错了!”。
第15 17行代码分别调用factory()方法获取对应的驱动对象。
从运行结果可以看出,工厂方法成功地创建了两个驱动类对象。