PHP的五种常见设计模式
工厂模式
最初在设计模式 一书中,许多设计模式都鼓励使用松散耦合。要理解这个概念,让我们最好谈一下许多开发人员从事大型系统的艰苦历程。在更改一个代码片段时,就会发生问题,系统其他部分 —— 您曾认为完全不相关的部分中也有可能出现级联破坏。该问题在于紧密耦合 。系统某个部分中的函数和类严重依赖于系统的其他部分中函数和类的行为和结构。您需要一组模式,使这些类能够相互通信,但不希望将它们紧密绑定在一起,以避免出现联锁。在大型系统中,许多代码依赖于少数几个关键类。需要更改这些类时,可能会出现困难。例如,假设您有一个从文件读取的 User
类。您希望将其更改为从数据库读取的其他类,但是,所有的代码都引用从文件读取的原始类。这时候,使用工厂模式会很方便。工厂模式 是一种类,它具有为您创建对象的某些方法。您可以使用工厂类创建对象,而不直接使用 new
。这样,如果您想要更改所创建的对象类型,只需更改该工厂即可。使用该工厂的所有代码会自动更改。清单 1 显示工厂类的一个示列。等式的服务器端包括两个部分:数据库和一组 PHP 页面,这些页面允许您添加反馈、请求反馈列表并获取与特定反馈相关的文章。
清单1:Factory1.php
1 <?php 2 interface IUser 3 { 4 function getName(); 5 } 6 class User implements IUser 7 { 8 public function __construct($id){} 9 10 public function getName() 11 { 12 return "Jack"; 13 } 14 } 15 class UserFactory 16 { 17 public static function Create($id) 18 { 19 return new User($id); 20 } 21 } 22 $uo = UserFactory::Create(1); 23 echo($uo->getName()."\n");
清单1的UML表示
测试代码会向工厂请求 User
对象,并输出 getName
方法的结果。
有一种工厂模式的变体使用工厂方法。类中的这些公共静态方法构造该类型的对象。如果创建此类型的对象非常重要,此方法非常有用。例如,假设您需要先创建对象,然后设置许多属性。此版本的工厂模式会将该进程封装在单个位置中,这样,不用复制复杂的初始化代码,也不必将复制好的代码在在代码库中到处粘贴。
清单2:Factory2.php
1 <?php 2 interface IUser 3 { 4 function getName(); 5 } 6 7 class User implements IUser 8 { 9 public static function Load($id) 10 { 11 return new User($id); 12 } 13 public static function Create() 14 { 15 return new User(null); 16 } 17 public function __construct($id){} 18 public function getName() 19 { 20 return "Jack2"; 21 } 22 } 23 $uo = User::Load(1); 24 echo($uo->getName()."\n");
清单2的RML表示
单元素模式
某些应用程序资源是独占的,因为有且只有一个此类型的资源。例如,通过数据库句柄到数据库的连接是独占的。您希望在应用程序中共享数据库句柄,因为在保持连接打开或关闭时,它是一种开销,在获取单个页面的过程中更是如此。
单元素模式可以满足此要求。如果应用程序每次包含且仅包含一个对象,那么这个对象就是一个单元素(Singleton)。清单 3 中的代码显示了 PHP V5 中的一个数据库连接单元素。
清单3:Singleton.php
1 <?php 2 require_once 'DB.php'; 3 4 class DatabaseConnection 5 { 6 public static function get() 7 { 8 static $db = null; 9 if ($db == null) 10 $db = new DatabaseConnection(); 11 return $db; 12 } 13 14 private $_handle = null; 15 16 private function __construct() 17 { 18 $dsn = 'mysql://root:password@localhost/photos'; 19 $this->_handle =&DB::Connect( $dsn, array() ); 20 } 21 public function handle() 22 { 23 return $this->_handle; 24 } 25 } 26 print("Handle = ".DatabaseConnection::get()->handle()."\n"); 27 print("Handle = ".DatabaseConnection::get()->handle()."\n");
清单3的UML表示
返回的两个句柄是同一对象。如果您在整个应用程序中使用数据库连接单元素,那么就可以在任何地方重用同一句柄。
您可以使用全局变量存储数据库句柄,但是,该方法仅适用于较小的应用程序。在较大的应用程序中,应避免使用全局变量,并使用对象和方法访问资源。
观察者模式
观察者模式为您提供了避免组件之间紧密耦合的另一种方法。该模式非常简单:一个对象通过添加一个方法(该方法允许另一个对象,即观察者 注册自己)使本身变得可观察。当可观察的对象更改时,它会将消息发送到已注册的观察者。这些观察者使用该信息执行的操作与可观察的对象无关。结果是对象可以相互对话,而不必了解原因。
一个简单示例是系统中的用户列表。清单 4 中的代码显示一个用户列表,添加用户时,它将发送出一条消息。添加用户时,通过发送消息的日志观察者可以观察此列表。
清单4:Observer.php
1 <?php 2 interface IObserver 3 { 4 function onChanged($sender,$args); 5 } 6 interface IObservable 7 { 8 function addObserver($observer); 9 } 10 class UserList implements IObservable 11 { 12 private $_observers = array(); 13 public function addCustomer($name) 14 { 15 foreach($this->_observers as $obs) 16 $obs->onChanged($this,$name); 17 } 18 public function addObserver($observer) 19 { 20 $this->_observers[] = $observer; 21 } 22 } 23 class UserListLogger implements IObserver 24 { 25 public function onChanged($sender,$args) 26 { 27 echo("$args added to user list\n"); 28 } 29 } 30 $ul = new UserList(); 31 $ul->addObserver(new UserListLogger()); 32 $ul->addCustomer("Jack");
清单4的UML表示
测试代码创建 UserList
,并将 UserListLogger
观察者添加到其中。然后添加一个消费者,并将这一更改通知 UserListLogger
。
认识到 UserList
不知道日志程序将执行什么操作很关键。可能存在一个或多个执行其他操作的侦听程序。例如,您可能有一个向新用户发送消息的观察者,欢迎新用户使用该系统。这种方法的价值在于 UserList
忽略所有依赖它的对象,它主要关注在列表更改时维护用户列表并发送消息这一工作。
此模式不限于内存中的对象。它是在较大的应用程序中使用的数据库驱动的消息查询系统的基础。
命令链模式
命令链模式以松散耦合主题为基础,发送消息、命令和请求,或通过一组处理程序发送任意内容。每个处理程序都会自行判断自己能否处理请求。如果可以,该请求被处理,进程停止。您可以为系统添加或移除处理程序,而不影响其他处理程序。清单 5 显示了此模式的一个示例。
清单5:Chain.php
1 <?php 2 interface ICommand 3 { 4 function onCommand($name,$args); 5 } 6 class CommandChain 7 { 8 private $_commands = array(); 9 public function addCommand($cmd) 10 { 11 $this->_commands[] = $cmd; 12 } 13 public function runCommand($name,$args) 14 { 15 foreach($this->_commands as $cmd) 16 { 17 if ($cmd->onCommand($name,$args)) 18 return; 19 } 20 } 21 } 22 class UserCommand implements ICommand 23 { 24 public function onCommand($name,$args) 25 { 26 if ($name != 'addUser') return false; 27 echo("UserCommand handling 'addUser'\n"); 28 return true; 29 } 30 } 31 class MailCommand implements ICommand 32 { 33 public function onCommand($name,$args) 34 { 35 if ($name != 'mail') return false; 36 echo("MailCommand handling 'mail'\n"); 37 return true; 38 } 39 } 40 41 $cc = new CommandChain(); 42 $cc->addCommand(new UserCommand()); 43 $cc->addCommand(new MailCommand()); 44 $cc->runCommand('addUser', null); 45 $cc->runCommand('mail', null);
清单5的UML表示
代码首先创建 CommandChain
对象,并为它添加两个命令对象的实例。然后运行两个命令以查看谁对这些命令作出了响应。如果命令的名称匹配 UserCommand
或 MailCommand
,则代码失败,不发生任何操作。
为处理请求而创建可扩展的架构时,命令链模式很有价值,使用它可以解决许多问题。
策略模式
我们讲述的最后一个设计模式是策略 模式。在此模式中,算法是从复杂类提取的,因而可以方便地替换。例如,如果要更改搜索引擎中排列页的方法,则策略模式是一个不错的选择。思考一下搜索引擎的几个部分 —— 一部分遍历页面,一部分对每页排列,另一部分基于排列的结果排序。在复杂的示例中,这些部分都在同一个类中。通过使用策略模式,您可将排列部分放入另一个类中,以便更改页排列的方式,而不影响搜索引擎的其余代码。
作为一个较简单的示例,清单 6 显示了一个用户列表类,它提供了一个根据一组即插即用的策略查找一组用户的方法。
清单6:Strategy.php
1 <?php 2 interface IStrategy 3 { 4 function filter($record); 5 } 6 class FindAfterStrategy implements IStrategy 7 { 8 private $_name; 9 public function __construct($name) 10 { 11 $this->_name = $name; 12 } 13 public function filter($record) 14 { 15 return strcmp($this->_name,$record) <= 0; 16 } 17 } 18 class RandomStrategy implements IStrategy 19 { 20 public function filter($record) 21 { 22 return rand(0,1) >= 0.5; 23 } 24 } 25 class UserList 26 { 27 private $_list = array(); 28 public function __construct($names) 29 { 30 if ($names != null) 31 { 32 foreach($names as $name) 33 $this->_list[] = $name; 34 } 35 } 36 public function add($name) 37 { 38 $this->_list[] = $name; 39 } 40 public function find($filter) 41 { 42 $recs = array(); 43 foreach($this->_list as $user) 44 { 45 if ($filter->filter($user)) 46 $recs[] = $user; 47 } 48 return $recs; 49 } 50 } 51 52 $ul = new UserList(array("Andy","Jack","Lori","Megan")); 53 $f1 = $ul->find(new FindAfterStrategy("J")); 54 print_r($f1); 55 $f2 = $ul->find(new RandomStrategy()); 56 print_r($f2);
清单6的UML表示
测试代码为两个策略运行同一用户列表,并显示结果。在第一种情况中,策略查找排列在 J
后的任何名称,所以您将得到 Jack、Lori 和 Megan。第二个策略随机选取名称,每次会产生不同的结果。在这种情况下,结果为 Andy 和 Megan。
策略模式非常适合复杂数据管理系统或数据处理系统,二者在数据筛选、搜索或处理的方式方面需要较高的灵活性。