首頁 php教程 PHP源码 五种常见的 PHP 设计模式

五种常见的 PHP 设计模式

May 25, 2016 pm 05:15 PM
php

工厂模式

最初在设计模式 一书中,许多设计模式都鼓励使用松散耦合。要理解这个概念,让我们最好谈一下许多开发人员从事大型系统的艰苦历程。在更改一个代码片段时,就会发生问题,系统其他部分 —— 您曾认为完全不相关的部分中也有可能出现级联破坏。

该问题在于紧密耦合 。系统某个部分中的函数和类严重依赖于系统的其他部分中函数和类的行为和结构。您需要一组模式,使这些类能够相互通信,但不希望将它们紧密绑定在一起,以避免出现联锁。

在大型系统中,许多代码依赖于少数几个关键类。需要更改这些类时,可能会出现困难。例如,假设您有一个从文件读取的 User 类。您希望将其更改为从数据库读取的其他类,但是,所有的代码都引用从文件读取的原始类。这时候,使用工厂模式会很方便。

工厂模式 是一种类,它具有为您创建对象的某些方法。您可以使用工厂类创建对象,而不直接使用 new。这样,如果您想要更改所创建的对象类型,只需更改该工厂即可。使用该工厂的所有代码会自动更改。

清单 1 显示工厂类的一个示列。等式的服务器端包括两个部分:数据库和一组 PHP 页面,这些页面允许您添加反馈、请求反馈列表并获取与特定反馈相关的文章。

<span style="font-family:Microsoft YaHei;font-size:14px;"><?php  
interface IUser  
{  
  function getName();  
}  
  
class User implements IUser  
{  
  public function __construct( $id ) { }  
  
  public function getName()  
  {  
    return "Jack";  
  }  
}  
  
class UserFactory  
{  
  public static function Create( $id )  
  {  
    return new User( $id );  
  }  
}  
  
$uo = UserFactory::Create( 1 );  
echo( $uo->getName()."\n" );  
?></span>
登入後複製

IUser 接口定义用户对象应执行什么操作。IUser 的实现称为 User,UserFactory 工厂类则创建 IUser 对象。此关系可以用图 1 中的 UML 表示。

图 1. 工厂类及其相关 IUser 接口和用户类

20160612120407805.gif

如果您使用 php 解释器在命令行上运行此代码,将得到如下结果:

<span style="font-family:Microsoft YaHei;font-size:14px;">% php factory1.php   
Jack  
%</span>
登入後複製

测试代码会向工厂请求 User 对象,并输出 getName 方法的结果。

有一种工厂模式的变体使用工厂方法。类中的这些公共静态方法构造该类型的对象。如果创建此类型的对象非常重要,此方法非常有用。例如,假设您需要先创建对象,然后设置许多属性。此版本的工厂模式会将该进程封装在单个位置中,这样,不用复制复杂的初始化代码,也不必将复制好的代码在在代码库中到处粘贴。

清单 2 显示使用工厂方法的一个示例。

<span style="font-family:Microsoft YaHei;font-size:14px;"><?php  
interface IUser  
{  
  function getName();  
}  
  
class User implements IUser  
{  
  public static function Load( $id )   
  {  
        return new User( $id );  
  }  
  
  public static function Create( )   
  {  
        return new User( null );  
  }  
  
  public function __construct( $id ) { }  
  
  public function getName()  
  {  
    return "Jack";  
  }  
}  
  
$uo = User::Load( 1 );  
echo( $uo->getName()."\n" );  
?></span>
登入後複製

这段代码要简单得多。它仅有一个接口 IUser 和一个实现此接口的 User 类。User 类有两个创建对象的静态方法。此关系可用图 2 中的 UML 表示。

图 2. IUser 接口和带有工厂方法的 user 类

20160612120538775.gif

在命令行中运行脚本产生的结果与清单 1 的结果相同,如下所示:

<span style="font-family:Microsoft YaHei;font-size:14px;">% php factory2.php   
Jack  
%</span>
登入後複製

如上所述,有时此类模式在规模较小的环境中似乎有些大材小用。不过,最好还是学习这种扎实的编码形式,以便应用于任意规模的项目中。

单元素模式

某些应用程序资源是独占的,因为有且只有一个此类型的资源。例如,通过数据库句柄到数据库的连接是独占的。您希望在应用程序中共享数据库句柄,因为在保持连接打开或关闭时,它是一种开销,在获取单个页面的过程中更是如此。

单元素模式可以满足此要求。如果应用程序每次包含且仅包含一个对象,那么这个对象就是一个单元素(Singleton)。清单 3 中的代码显示了 PHP V5 中的一个数据库连接单元素。

清单 3. Singleton.php

<span style="font-family:Microsoft YaHei;font-size:14px;"><?php  
require_once("DB.php");  
  
class DatabaseConnection  
{  
  public static function get()  
  {  
    static $db = null;  
    if ( $db == null )  
      $db = new DatabaseConnection();  
    return $db;  
  }  
  
  private $_handle = null;  
  
  private function __construct()  
  {  
    $dsn = &#39;mysql://root:password@localhost/photos&#39;;  
    $this->_handle =& DB::Connect( $dsn, array() );  
  }  
    
  public function handle()  
  {  
    return $this->_handle;  
  }  
}  
  
print( "Handle = ".DatabaseConnection::get()->handle()."\n" );  
print( "Handle = ".DatabaseConnection::get()->handle()."\n" );  
?></span>
登入後複製

此代码显示名为 DatabaseConnection 的单个类。您不能创建自已的 DatabaseConnection,因为构造函数是专用的。但使用静态 get 方法,您可以获得且仅获得一个 DatabaseConnection 对象。此代码的 UML 如图 3 所示。

图 3. 数据库连接单元素

20160612120652660.gif

在两次调用间,handle 方法返回的数据库句柄是相同的,这就是最好的证明。您可以在命令行中运行代码来观察这一点。

<span style="font-family:Microsoft YaHei;font-size:14px;">% php singleton.php   
Handle = Object id #3  
Handle = Object id #3  
%</span>
登入後複製

返回的两个句柄是同一对象。如果您在整个应用程序中使用数据库连接单元素,那么就可以在任何地方重用同一句柄。

您可以使用全局变量存储数据库句柄,但是,该方法仅适用于较小的应用程序。在较大的应用程序中,应避免使用全局变量,并使用对象和方法访问资源。

观察者模式

观察者模式为您提供了避免组件之间紧密耦合的另一种方法。该模式非常简单:一个对象通过添加一个方法(该方法允许另一个对象,即观察者 注册自己)使本身变得可观察。当可观察的对象更改时,它会将消息发送到已注册的观察者。这些观察者使用该信息执行的操作与可观察的对象无关。结果是对象可以相互对话,而不必了解原因。

一个简单示例是系统中的用户列表。清单 4 中的代码显示一个用户列表,添加用户时,它将发送出一条消息。添加用户时,通过发送消息的日志观察者可以观察此列表。

清单 4. Observer.php

<span style="font-family:Microsoft YaHei;font-size:14px;"><?php  
interface IObserver  
{  
  function onChanged( $sender, $args );  
}  
  
interface IObservable  
{  
  function addObserver( $observer );  
}  
  
class UserList implements IObservable  
{  
  private $_observers = array();  
  
  public function addCustomer( $name )  
  {  
    foreach( $this->_observers as $obs )  
      $obs->onChanged( $this, $name );  
  }  
  
  public function addObserver( $observer )  
  {  
    $this->_observers []= $observer;  
  }  
}  
  
class UserListLogger implements IObserver  
{  
  public function onChanged( $sender, $args )  
  {  
    echo( "&#39;$args&#39; added to user list\n" );  
  }  
}  
  
$ul = new UserList();  
$ul->addObserver( new UserListLogger() );  
$ul->addCustomer( "Jack" );  
?></span>
登入後複製

此代码定义四个元素:两个接口和两个类。IObservable 接口定义可以被观察的对象,UserList 实现该接口,以便将本身注册为可观察。IObserver 列表定义要通过怎样的方法才能成为观察者,UserListLogger 实现 IObserver 接口。图 4 的 UML 中展示了这些元素。

图 4. 可观察的用户列表和用户列表事件日志程序

20160612120757693.gif

如果在命令行中运行它,您将看到以下输出:

<span style="font-family:Microsoft YaHei;font-size:14px;">% php observer.php   
&#39;Jack&#39; added to user list  
%</span>
登入後複製

测试代码创建 UserList,并将 UserListLogger 观察者添加到其中。然后添加一个消费者,并将这一更改通知 UserListLogger。

认识到 UserList 不知道日志程序将执行什么操作很关键。可能存在一个或多个执行其他操作的侦听程序。例如,您可能有一个向新用户发送消息的观察者,欢迎新用户使用该系统。这种方法的价值在于 UserList 忽略所有依赖它的对象,它主要关注在列表更改时维护用户列表并发送消息这一工作。

此模式不限于内存中的对象。它是在较大的应用程序中使用的数据库驱动的消息查询系统的基础。

命令链模式

命令链 模式以松散耦合主题为基础,发送消息、命令和请求,或通过一组处理程序发送任意内容。每个处理程序都会自行判断自己能否处理请求。如果可以,该请求被处理,进程停止。您可以为系统添加或移除处理程序,而不影响其他处理程序。清单 5 显示了此模式的一个示例。

清单 5. Chain.php

<span style="font-family:Microsoft YaHei;font-size:14px;"><?php  
interface ICommand  
{  
  function onCommand( $name, $args );  
}  
  
class CommandChain  
{  
  private $_commands = array();  
  
  public function addCommand( $cmd )  
  {  
    $this->_commands []= $cmd;  
  }  
  
  public function runCommand( $name, $args )  
  {  
    foreach( $this->_commands as $cmd )  
    {  
      if ( $cmd->onCommand( $name, $args ) )  
        return;  
    }  
  }  
}  
  
class UserCommand implements ICommand  
{  
  public function onCommand( $name, $args )  
  {  
    if ( $name != &#39;addUser&#39; ) return false;  
    echo( "UserCommand handling &#39;addUser&#39;\n" );  
    return true;  
  }  
}  
  
class MailCommand implements ICommand  
{  
  public function onCommand( $name, $args )  
  {  
    if ( $name != &#39;mail&#39; ) return false;  
    echo( "MailCommand handling &#39;mail&#39;\n" );  
    return true;  
  }  
}  
  
$cc = new CommandChain();  
$cc->addCommand( new UserCommand() );  
$cc->addCommand( new MailCommand() );  
$cc->runCommand( &#39;addUser&#39;, null );  
$cc->runCommand( &#39;mail&#39;, null );  
?></span>
登入後複製

此代码定义维护 ICommand 对象列表的 CommandChain 类。两个类都可以实现 ICommand 接口 —— 一个对邮件的请求作出响应,另一个对添加用户作出响应。 图 5 给出了 UML。

图 5. 命令链及其相关命令

20160612120859432.gif

如果您运行包含某些测试代码的脚本,则会得到以下输出:

<span style="font-family:Microsoft YaHei;font-size:14px;">% php chain.php   
UserCommand handling &#39;addUser&#39;  
MailCommand handling &#39;mail&#39;  
%</span>
登入後複製

代码首先创建 CommandChain 对象,并为它添加两个命令对象的实例。然后运行两个命令以查看谁对这些命令作出了响应。如果命令的名称匹配 UserCommand 或 MailCommand,则代码失败,不发生任何操作。

为处理请求而创建可扩展的架构时,命令链模式很有价值,使用它可以解决许多问题。

策略模式

我们讲述的最后一个设计模式是策略 模式。在此模式中,算法是从复杂类提取的,因而可以方便地替换。例如,如果要更改搜索引擎中排列页的方法,则策略模式是一个不错的选择。思考一下搜索引擎的几个部分 —— 一部分遍历页面,一部分对每页排列,另一部分基于排列的结果排序。在复杂的示例中,这些部分都在同一个类中。通过使用策略模式,您可将排列部分放入另一个类中,以便更改页排列的方式,而不影响搜索引擎的其余代码。

作为一个较简单的示例,清单 6 显示了一个用户列表类,它提供了一个根据一组即插即用的策略查找一组用户的方法。

清单 6. Strategy.php

<span style="font-family:Microsoft YaHei;font-size:14px;"><?php  
interface IStrategy  
{  
  function filter( $record );  
}  
  
class FindAfterStrategy implements IStrategy  
{  
  private $_name;  
  
  public function __construct( $name )  
  {  
    $this->_name = $name;  
  }  
  
  public function filter( $record )  
  {  
    return strcmp( $this->_name, $record ) <= 0;  
  }  
}  
  
class RandomStrategy implements IStrategy  
{  
  public function filter( $record )  
  {  
    return rand( 0, 1 ) >= 0.5;  
  }  
}  
  
class UserList  
{  
  private $_list = array();  
  
  public function __construct( $names )  
  {  
    if ( $names != null )  
    {  
      foreach( $names as $name )  
      {  
        $this->_list []= $name;  
      }  
    }  
  }  
  
  public function add( $name )  
  {  
    $this->_list []= $name;  
  }  
  
  public function find( $filter )  
  {  
    $recs = array();  
    foreach( $this->_list as $user )  
    {  
      if ( $filter->filter( $user ) )  
        $recs []= $user;  
    }  
    return $recs;  
  }  
}  
  
$ul = new UserList( array( "Andy", "Jack", "Lori", "Megan" ) );  
$f1 = $ul->find( new FindAfterStrategy( "J" ) );  
print_r( $f1 );  
  
$f2 = $ul->find( new RandomStrategy() );  
print_r( $f2 );  
?></span>
登入後複製

此代码的 UML 如图 6 所示。

图 6. 用户列表和用于选择用户的策略

20160612121012931.gif

UserList 类是打包名称数组的一个包装器。它实现 find 方法,该方法利用几个策略之一来选择这些名称的子集。这些策略由 IStrategy 接口定义,该接口有两个实现:一个随机选择用户,另一个根据指定名称选择其后的所有名称。运行测试代码时,将得到以下输出:

<span style="font-family:Microsoft YaHei;font-size:14px;">% php strategy.php   
Array  
(  
    [0] => Jack  
    [1] => Lori  
    [2] => Megan  
)  
Array  
(  
    [0] => Andy  
    [1] => Megan  
)  
%</span>
登入後複製

测试代码为两个策略运行同一用户列表,并显示结果。在第一种情况中,策略查找排列在 J 后的任何名称,所以您将得到 Jack、Lori 和 Megan。第二个策略随机选取名称,每次会产生不同的结果。在这种情况下,结果为 Andy 和 Megan。

策略模式非常适合复杂数据管理系统或数据处理系统,二者在数据筛选、搜索或处理的方式方面需要较高的灵活性。

结束语

本文介绍的仅仅是 PHP 应用程序中使用的几种最常见的设计模式。在设计模式 一书中演示了更多的设计模式。不要因架构的神秘性而放弃。模式是一种绝妙的理念,适用于任何编程语言、任何技能水平。


本網站聲明
本文內容由網友自願投稿,版權歸原作者所有。本站不承擔相應的法律責任。如發現涉嫌抄襲或侵權的內容,請聯絡admin@php.cn

熱AI工具

Undresser.AI Undress

Undresser.AI Undress

人工智慧驅動的應用程序,用於創建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用於從照片中去除衣服的線上人工智慧工具。

Undress AI Tool

Undress AI Tool

免費脫衣圖片

Clothoff.io

Clothoff.io

AI脫衣器

Video Face Swap

Video Face Swap

使用我們完全免費的人工智慧換臉工具,輕鬆在任何影片中換臉!

熱門文章

<🎜>:泡泡膠模擬器無窮大 - 如何獲取和使用皇家鑰匙
3 週前 By 尊渡假赌尊渡假赌尊渡假赌
北端:融合系統,解釋
3 週前 By 尊渡假赌尊渡假赌尊渡假赌

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的程式碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

強大的PHP整合開發環境

Dreamweaver CS6

Dreamweaver CS6

視覺化網頁開發工具

SublimeText3 Mac版

SublimeText3 Mac版

神級程式碼編輯軟體(SublimeText3)

熱門話題

Java教學
1664
14
CakePHP 教程
1423
52
Laravel 教程
1318
25
PHP教程
1268
29
C# 教程
1248
24
在PHP API中說明JSON Web令牌(JWT)及其用例。 在PHP API中說明JSON Web令牌(JWT)及其用例。 Apr 05, 2025 am 12:04 AM

JWT是一種基於JSON的開放標準,用於在各方之間安全地傳輸信息,主要用於身份驗證和信息交換。 1.JWT由Header、Payload和Signature三部分組成。 2.JWT的工作原理包括生成JWT、驗證JWT和解析Payload三個步驟。 3.在PHP中使用JWT進行身份驗證時,可以生成和驗證JWT,並在高級用法中包含用戶角色和權限信息。 4.常見錯誤包括簽名驗證失敗、令牌過期和Payload過大,調試技巧包括使用調試工具和日誌記錄。 5.性能優化和最佳實踐包括使用合適的簽名算法、合理設置有效期、

php程序在字符串中計數元音 php程序在字符串中計數元音 Feb 07, 2025 pm 12:12 PM

字符串是由字符組成的序列,包括字母、數字和符號。本教程將學習如何使用不同的方法在PHP中計算給定字符串中元音的數量。英語中的元音是a、e、i、o、u,它們可以是大寫或小寫。 什麼是元音? 元音是代表特定語音的字母字符。英語中共有五個元音,包括大寫和小寫: a, e, i, o, u 示例 1 輸入:字符串 = "Tutorialspoint" 輸出:6 解釋 字符串 "Tutorialspoint" 中的元音是 u、o、i、a、o、i。總共有 6 個元

解釋PHP中的晚期靜態綁定(靜態::)。 解釋PHP中的晚期靜態綁定(靜態::)。 Apr 03, 2025 am 12:04 AM

靜態綁定(static::)在PHP中實現晚期靜態綁定(LSB),允許在靜態上下文中引用調用類而非定義類。 1)解析過程在運行時進行,2)在繼承關係中向上查找調用類,3)可能帶來性能開銷。

什麼是PHP魔術方法(__ -construct,__destruct,__call,__get,__ set等)並提供用例? 什麼是PHP魔術方法(__ -construct,__destruct,__call,__get,__ set等)並提供用例? Apr 03, 2025 am 12:03 AM

PHP的魔法方法有哪些? PHP的魔法方法包括:1.\_\_construct,用於初始化對象;2.\_\_destruct,用於清理資源;3.\_\_call,處理不存在的方法調用;4.\_\_get,實現動態屬性訪問;5.\_\_set,實現動態屬性設置。這些方法在特定情況下自動調用,提升代碼的靈活性和效率。

PHP和Python:比較兩種流行的編程語言 PHP和Python:比較兩種流行的編程語言 Apr 14, 2025 am 12:13 AM

PHP和Python各有優勢,選擇依據項目需求。 1.PHP適合web開發,尤其快速開發和維護網站。 2.Python適用於數據科學、機器學習和人工智能,語法簡潔,適合初學者。

PHP行動:現實世界中的示例和應用程序 PHP行動:現實世界中的示例和應用程序 Apr 14, 2025 am 12:19 AM

PHP在電子商務、內容管理系統和API開發中廣泛應用。 1)電子商務:用於購物車功能和支付處理。 2)內容管理系統:用於動態內容生成和用戶管理。 3)API開發:用於RESTfulAPI開發和API安全性。通過性能優化和最佳實踐,PHP應用的效率和可維護性得以提升。

PHP:網絡開發的關鍵語言 PHP:網絡開發的關鍵語言 Apr 13, 2025 am 12:08 AM

PHP是一種廣泛應用於服務器端的腳本語言,特別適合web開發。 1.PHP可以嵌入HTML,處理HTTP請求和響應,支持多種數據庫。 2.PHP用於生成動態網頁內容,處理表單數據,訪問數據庫等,具有強大的社區支持和開源資源。 3.PHP是解釋型語言,執行過程包括詞法分析、語法分析、編譯和執行。 4.PHP可以與MySQL結合用於用戶註冊系統等高級應用。 5.調試PHP時,可使用error_reporting()和var_dump()等函數。 6.優化PHP代碼可通過緩存機制、優化數據庫查詢和使用內置函數。 7

PHP的持久相關性:它還活著嗎? PHP的持久相關性:它還活著嗎? Apr 14, 2025 am 12:12 AM

PHP仍然具有活力,其在現代編程領域中依然佔據重要地位。 1)PHP的簡單易學和強大社區支持使其在Web開發中廣泛應用;2)其靈活性和穩定性使其在處理Web表單、數據庫操作和文件處理等方面表現出色;3)PHP不斷進化和優化,適用於初學者和經驗豐富的開發者。

See all articles