浅谈PHP面向对象编程(九、设计模式)

时间:2023-03-09 09:53:37
浅谈PHP面向对象编程(九、设计模式)

9.0 设计模式


  在编写程序时经常会遇到一此典型的问题或需要完成某种特定需求,设计模式就是针对这些问题和需求,在大量的实践中总结和理论化之后优选的代码结构编程风格,以及解决问题的思考方式。

设计模式就像是经典的棋谱。不同的棋局,使用不同的棋谐,免得自己再去思考和模索。本节将针对PHP应用程序中最常用的两种设计模式进行详细讲解。

9.1 单例模式


  单例模式是PHP中的一种设计模式,它是指在设计一个类时,需要保证在整个程序运行期间针对该类只存在一个实例对象。

就像世界上只有一个月亮,假设现在要设计一个类表示月亮,该类只能有一个实例对象,否则就违背了事实。

在讲解单例设计模式之前,通过一个案例来演示在什么情况时需要使用单例模式,如例9-23 所示

例9-2

 <?php
class dbHelper{
private $conn = null;
public function __construct(){
//打开一个到 MySQL 服务器的连接
$this->conn = mysql_connect("localhost","root","");
echo "得到一个conn<br/>";
}
}
$db1 = new dbHelper();
$db2 = new dbHelper();
if($db1 === $db2){
echo "一个对象<br/>";
} else {
echo "两个对象<br/>";
}
?>

运行结果

得到一个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文件。示例代码如下:

 <?php
class MySQL{ //操作SQL的驱动类 }

然后在根目录下创建SQLite.php文件。示例代码如下:

 <?php
class SQLite{ //操作SQLite的驱动类 }

最后定义一个工厂方法来获取各驱动对象,代码如例9-25所示

例9-25

 <?php
header('Content-Type: textml;charset=utf-8');
class Db{
//工厂方法
public static function factory($type){
if (include_once $type . '.php') {
$classname = $type;
return new $classname();
} else {
echo "出错了!";
}
}
}
//获取MySQL驱动对象
$mysql = Db::factory('MySQL');
//获取SQLite驱动对象
$sqlite = Db::factory('SQLite');
var_dump($mysql);
var_dump($sqlite);
?>

运行结果

object (MySQL) [1]

object (SQLite) [2]

上例中,第5行代码定义了一个静态方法factor(), 这就是工厂方法,该方法的参数为类名。

第6- 11行代码用于判断类名与参数是否相同,如果相同则创建该类的对象,否则输出“出错了!”。

第15 17行代码分别调用factory()方法获取对应的驱动对象。

从运行结果可以看出,工厂方法成功地创建了两个驱动类对象。