国产av日韩一区二区三区精品,成人性爱视频在线观看,国产,欧美,日韩,一区,www.成色av久久成人,2222eeee成人天堂

首頁 后端開發(fā) php教程 PHP主|零對象模式 - 域模型中的多態(tài)性

PHP主|零對象模式 - 域模型中的多態(tài)性

Feb 25, 2025 pm 02:53 PM

PHP Master | The Null Object Pattern - Polymorphism in Domain Models

核心要點

  • 空對象模式是一種設計模式,利用多態(tài)性減少條件代碼,使代碼更簡潔易維護。它提供一個非功能性對象,可以替代真實對象,從而無需進行空值檢查。
  • 空對象模式可以與其他設計模式結合使用,例如工廠模式創(chuàng)建和返回空對象,或策略模式在運行時更改對象的行為。
  • 空對象模式的潛在缺點是可能導致創(chuàng)建不必要的對象,增加內(nèi)存使用。它也可能使代碼更復雜,因為需要創(chuàng)建額外的類和接口。
  • 實現(xiàn)空對象模式需要創(chuàng)建一個空對象類,該類實現(xiàn)與真實對象相同的接口。這個空對象為接口中的所有方法提供默認實現(xiàn),允許它替代真實對象。這使得代碼更健壯,更不容易出錯。

雖然并非完全符合規(guī)范,但可以說正交性是基于“良好設計”原則的軟件系統(tǒng)的精髓,其中各個模塊彼此解耦,使系統(tǒng)不易出現(xiàn)僵化和脆弱的問題。當然,談論正交系統(tǒng)的優(yōu)點比實際運行生產(chǎn)系統(tǒng)更容易;這個過程通常是追求烏托邦。即使如此,在系統(tǒng)中實現(xiàn)高度解耦的組件絕非烏托邦的概念。多態(tài)性等多種編程概念允許設計靈活的程序,其各個部分可以在運行時切換,其依賴關系可以以抽象的形式表達,而不是具體的實現(xiàn)。我想說,無論我們是在實現(xiàn)基礎設施還是應用程序邏輯,“面向接口編程”的舊格言隨著時間的推移已經(jīng)得到了普遍的采用。然而,當踏上領域模型的領域時,情況就大相徑庭了。坦率地說,這是一個可預測的場景。畢竟,為什么一個相互關聯(lián)的對象網(wǎng)絡(其數(shù)據(jù)和行為受限于明確定義的業(yè)務規(guī)則)應該是多態(tài)的呢?它本身并沒有多大意義。不過,此規(guī)則有一些例外,最終可能適用于這種情況。第一個例外是使用虛擬代理,它有效地與實際領域對象實現(xiàn)的接口相同。另一個例外是所謂的“空情況”,這是一種特殊情況,其中操作最終可能分配或返回空值,而不是填充良好的實體。在傳統(tǒng)的非多態(tài)方法中,模型的使用者必須檢查這些“有害的”空值并優(yōu)雅地處理該條件,從而在整個代碼中產(chǎn)生條件語句的爆炸。幸運的是,只需創(chuàng)建領域對象的多分支實現(xiàn)即可輕松解決這種看似混亂的情況,該實現(xiàn)將實現(xiàn)與所討論對象相同的接口,只是其方法不會執(zhí)行任何操作,因此將客戶端代碼從在執(zhí)行操作時重復檢查難看的空值中解脫出來。毫不奇怪,這種方法是一種稱為空對象的設計模式,它將多態(tài)性的優(yōu)勢發(fā)揮到極致。在本文中,我將演示該模式在幾種情況下的好處,并向您展示它們?nèi)绾尉o密地依附于多態(tài)方法。

處理非多態(tài)條件

正如人們所料,在展示空對象模式的優(yōu)點時,有幾種方法可以嘗試。我發(fā)現(xiàn)一種特別直接的方法是實現(xiàn)一個數(shù)據(jù)映射器,它最終可能從通用查找器返回空值。假設我們已經(jīng)成功創(chuàng)建了一個骨架領域模型,該模型僅由一個用戶實體組成。接口及其類如下所示:

<?php namespace Model;

interface UserInterface
{
    public function setId($id);
    public function getId();

    public function setName($name);
    public function getName();

    public function setEmail($email);
    public function getEmail();
}
<?php namespace Model;

class User implements UserInterface
{
    private $id;
    private $name;
    private $email;

    public function __construct($name, $email) {
        $this->setName($name);
        $this->setEmail($email);
    }

    public function setId($id) {
        if ($this->id !== null) {
            throw new BadMethodCallException(
                "The ID for this user has been set already.");
        }
        if (!is_int($id) || $id             throw new InvalidArgumentException(
              "The ID for this user is invalid.");
        }
        $this->id = $id;
        return $this;
    }

    public function getId() {
        return $this->id;
    }

    public function setName($name) {
        if (strlen($name)  30) {
            throw new InvalidArgumentException(
                "The user name is invalid.");
        }
        $this->name = $name;
        return $this;
    }

    public function getName() {
        return $this->name;
    }

    public function setEmail($email) {
        if (!filter_var($email, FILTER_VALIDATE_EMAIL)) {
            throw new InvalidArgumentException(
                "The user email is invalid.");
        }
        $this->email = $email;
        return $this;
    }

    public function getEmail() {
        return $this->email;
    }
}

User 類是一個反應式結構,它實現(xiàn)了一些 mutators/accessors 來定義一些用戶的數(shù)據(jù)和行為。有了這個構造的領域類,我們現(xiàn)在可以更進一步,定義一個基本的數(shù)據(jù)映射器,它將我們的領域模型和數(shù)據(jù)訪問層彼此隔離。

<?php namespace ModelMapper;
use LibraryDatabaseDatabaseAdapterInterface,
    ModelUser;

class UserMapper implements UserMapperInterface
{   
    private $adapter;

    public function __construct(DatabaseAdapterInterface $adapter) {
        $this->adapter = $adapter;
    }

    public function fetchById($id) {
        $this->adapter->select("users", array("id" => $id));
        if (!$row = $this->adapter->fetch()) {
            return null;
        }
        return $this->createUser($row);
    }

    private function createUser(array $row) {
        $user = new User($row["name"], $row["email"]);
        $user->setId($row["id"]);
        return $user;
    }
}

首先應該出現(xiàn)的是映射器的 fetchById() 方法是塊中的淘氣鬼,因為它在數(shù)據(jù)庫中沒有用戶與給定的 ID 匹配時有效地返回 null。出于顯而易見的原因,這個笨拙的條件使得客戶端代碼每次調(diào)用映射器的查找器時都必須費力地檢查空值。

<?php use LibraryLoaderAutoloader,
    LibraryDatabasePdoAdapter,
    ModelMapperUserMapper;

require_once __DIR__ . "/Library/Loader/Autoloader.php";
$autoloader = new Autoloader;
$autoloader->register();

$adapter = new PdoAdapter("mysql:dbname=test", "myusername", "mypassword");

$userMapper = new UserMapper($adapter);

$user = $userMapper->fetchById(1);

if ($user !== null) {
    echo $user->getName() . " " . $user->getEmail();
}

乍一看,這不會有什么問題,只要在一個地方進行檢查即可。但是,如果相同的行出現(xiàn)在多個頁面控制器或服務層中,您會不會撞到磚墻上?在你意識到之前,映射器返回的看似無辜的 null 會產(chǎn)生大量重復的條件,這是設計不良的兇兆。

從客戶端代碼中刪除條件語句

然而,無需焦慮,因為這正是空對象模式顯示多態(tài)性為何是天賜之物的情況。如果我們想一勞永逸地擺脫那些討厭的條件語句,我們可以實現(xiàn)前面 User 類的多態(tài)版本。

<?php namespace Model;

class NullUser implements UserInterface
{
    public function setId($id) { }
    public function getId() { }

    public function setName($name) { }
    public function getName() { }

    public function setEmail($email) { }
    public function getEmail() { }
}

如果您期望一個完整的實體類打包各種各樣的裝飾,那么恐怕您會非常失望。“null”版本的實體符合相應的接口,但方法是空包裝器,沒有實際的實現(xiàn)。雖然 NullUser 類的存在顯然沒有給我們帶來任何值得稱贊的有用東西,但它是一個簡潔的結構,允許我們將所有之前的條件語句扔進垃圾桶。想看看它是如何實現(xiàn)的嗎?首先,我們應該做一些前期工作并重構數(shù)據(jù)映射器,以便它的查找器返回一個空用戶對象而不是空值。

<?php namespace ModelMapper;
use LibraryDatabaseDatabaseAdapterInterface,
    ModelUser,
    ModelNullUser;

class UserMapper implements UserMapperInterface
{   
    private $adapter;

    public function __construct(DatabaseAdapterInterface $adapter) {
        $this->adapter = $adapter;
    }

    public function fetchById($id) {
        $this->adapter->select("users", array("id" => $id));
        return $this->createUser($this->adapter->fetch());
    }

    private function createUser($row) {
        if (!$row) {
            return new NullUser;
        }
        $user = new User($row["name"], $row["email"]);
        $user->setId($row["id"]);
        return $user; 
    }
}

映射器的 createUser() 方法隱藏了一個微小的條件,因為它現(xiàn)在負責在傳遞給查找器的 ID 沒有返回有效用戶時創(chuàng)建一個空用戶。即便如此,這種細微的代價不僅可以節(jié)省客戶端代碼進行大量重復檢查的工作,而且還可以將其變成一個寬松的使用者,當它必須處理空用戶時不會抱怨。

<?php namespace Model;

interface UserInterface
{
    public function setId($id);
    public function getId();

    public function setName($name);
    public function getName();

    public function setEmail($email);
    public function getEmail();
}

這種多態(tài)方法的主要缺點是,任何使用它的應用程序都將變得過于寬松,因為它在處理無效實體時永遠不會崩潰。在最壞的情況下,用戶界面只會顯示一些空白行,但沒有任何真正嘈雜的東西讓我們感到厭惡。當掃描早期 NullUser 類的當前實現(xiàn)時,這一點尤其明顯。即使是可行的,更不用說推薦的了,在保持其多態(tài)性不變的同時,也可以在空對象中封裝邏輯。我什至可以說,空對象非常適合封裝默認數(shù)據(jù)和行為,這些數(shù)據(jù)和行為應該只在少數(shù)特殊情況下向客戶端代碼公開。如果您足夠雄心勃勃并且想使用簡單的空用戶對象嘗試這個概念,那么當前的 NullUser 類可以按以下方式重構:

<?php namespace Model;

class User implements UserInterface
{
    private $id;
    private $name;
    private $email;

    public function __construct($name, $email) {
        $this->setName($name);
        $this->setEmail($email);
    }

    public function setId($id) {
        if ($this->id !== null) {
            throw new BadMethodCallException(
                "The ID for this user has been set already.");
        }
        if (!is_int($id) || $id             throw new InvalidArgumentException(
              "The ID for this user is invalid.");
        }
        $this->id = $id;
        return $this;
    }

    public function getId() {
        return $this->id;
    }

    public function setName($name) {
        if (strlen($name)  30) {
            throw new InvalidArgumentException(
                "The user name is invalid.");
        }
        $this->name = $name;
        return $this;
    }

    public function getName() {
        return $this->name;
    }

    public function setEmail($email) {
        if (!filter_var($email, FILTER_VALIDATE_EMAIL)) {
            throw new InvalidArgumentException(
                "The user email is invalid.");
        }
        $this->email = $email;
        return $this;
    }

    public function getEmail() {
        return $this->email;
    }
}

增強的 NullUser 版本比其安靜的前身略微更具表達性,因為它的 getter 提供了一些基本的實現(xiàn),以便在請求無效用戶時返回一些默認消息。雖然微不足道,但這項更改對客戶端代碼處理空用戶的方式產(chǎn)生了積極的影響,因為這次使用者至少清楚地知道當他們試圖從存儲中提取不存在的用戶時出現(xiàn)了一些問題。這是一個不錯的突破,它不僅展示了如何實現(xiàn)實際上根本不是空的空對象,還展示了根據(jù)特定需求在相關對象內(nèi)部移動邏輯是多么容易。

結束語

有些人可能會說,實現(xiàn)空對象很麻煩,尤其是在 PHP 中,OOP 的核心概念(如多態(tài)性)被明顯低估了。他們在某種程度上是對的。盡管如此,逐步采用值得信賴的編程原則和設計模式,以及該語言對象模型目前達到的成熟度水平,為穩(wěn)步前進和開始使用一些不久前被認為是復雜、不切實際的概念的“奢侈品”提供了所有必要的基礎??諏ο竽J綄儆诖祟悇e,但其實現(xiàn)非常簡單和優(yōu)雅,以至于在清除客戶端代碼中的重復空值檢查時很難不覺得它很吸引人。 圖片來自 Fotolia

(由于篇幅限制,此處省略了原文中的FAQ部分。)

以上是PHP主|零對象模式 - 域模型中的多態(tài)性的詳細內(nèi)容。更多信息請關注PHP中文網(wǎng)其他相關文章!

本站聲明
本文內(nèi)容由網(wǎng)友自發(fā)貢獻,版權歸原作者所有,本站不承擔相應法律責任。如您發(fā)現(xiàn)有涉嫌抄襲侵權的內(nèi)容,請聯(lián)系admin@php.cn

熱AI工具

Undress AI Tool

Undress AI Tool

免費脫衣服圖片

Undresser.AI Undress

Undresser.AI Undress

人工智能驅動的應用程序,用于創(chuàng)建逼真的裸體照片

AI Clothes Remover

AI Clothes Remover

用于從照片中去除衣服的在線人工智能工具。

Clothoff.io

Clothoff.io

AI脫衣機

Video Face Swap

Video Face Swap

使用我們完全免費的人工智能換臉工具輕松在任何視頻中換臉!

熱工具

記事本++7.3.1

記事本++7.3.1

好用且免費的代碼編輯器

SublimeText3漢化版

SublimeText3漢化版

中文版,非常好用

禪工作室 13.0.1

禪工作室 13.0.1

功能強大的PHP集成開發(fā)環(huán)境

Dreamweaver CS6

Dreamweaver CS6

視覺化網(wǎng)頁開發(fā)工具

SublimeText3 Mac版

SublimeText3 Mac版

神級代碼編輯軟件(SublimeText3)

熱門話題

Laravel 教程
1601
29
PHP教程
1502
276
PHP變量范圍解釋了 PHP變量范圍解釋了 Jul 17, 2025 am 04:16 AM

PHP變量作用域常見問題及解決方法包括:1.函數(shù)內(nèi)部無法訪問全局變量,需使用global關鍵字或參數(shù)傳入;2.靜態(tài)變量用static聲明,只初始化一次并在多次調(diào)用間保持值;3.超全局變量如$_GET、$_POST可在任何作用域直接使用,但需注意安全過濾;4.匿名函數(shù)需通過use關鍵字引入父作用域變量,修改外部變量則需傳遞引用。掌握這些規(guī)則有助于避免錯誤并提升代碼穩(wěn)定性。

如何在PHP中牢固地處理文件上傳? 如何在PHP中牢固地處理文件上傳? Jul 08, 2025 am 02:37 AM

要安全處理PHP文件上傳需驗證來源與類型、控制文件名與路徑、設置服務器限制并二次處理媒體文件。1.驗證上傳來源通過token防止CSRF并通過finfo_file檢測真實MIME類型使用白名單控制;2.重命名文件為隨機字符串并根據(jù)檢測類型決定擴展名存儲至非Web目錄;3.PHP配置限制上傳大小及臨時目錄Nginx/Apache禁止訪問上傳目錄;4.GD庫重新保存圖片清除潛在惡意數(shù)據(jù)。

在PHP中評論代碼 在PHP中評論代碼 Jul 18, 2025 am 04:57 AM

PHP注釋代碼常用方法有三種:1.單行注釋用//或#屏蔽一行代碼,推薦使用//;2.多行注釋用/.../包裹代碼塊,不可嵌套但可跨行;3.組合技巧注釋如用/if(){}/控制邏輯塊,或配合編輯器快捷鍵提升效率,使用時需注意閉合符號和避免嵌套。

發(fā)電機如何在PHP中工作? 發(fā)電機如何在PHP中工作? Jul 11, 2025 am 03:12 AM

AgeneratorinPHPisamemory-efficientwaytoiterateoverlargedatasetsbyyieldingvaluesoneatatimeinsteadofreturningthemallatonce.1.Generatorsusetheyieldkeywordtoproducevaluesondemand,reducingmemoryusage.2.Theyareusefulforhandlingbigloops,readinglargefiles,or

撰寫PHP評論的提示 撰寫PHP評論的提示 Jul 18, 2025 am 04:51 AM

寫好PHP注釋的關鍵在于明確目的與規(guī)范,注釋應解釋“為什么”而非“做了什么”,避免冗余或過于簡單。1.使用統(tǒng)一格式,如docblock(/*/)用于類、方法說明,提升可讀性與工具兼容性;2.強調(diào)邏輯背后的原因,如說明為何需手動輸出JS跳轉;3.在復雜代碼前添加總覽性說明,分步驟描述流程,幫助理解整體思路;4.合理使用TODO和FIXME標記待辦事項與問題,便于后續(xù)追蹤與協(xié)作。好的注釋能降低溝通成本,提升代碼維護效率。

學習PHP:初學者指南 學習PHP:初學者指南 Jul 18, 2025 am 04:54 AM

易于效率,啟動啟動tingupalocalserverenverenvirestoolslikexamppandacodeeditorlikevscode.1)installxamppforapache,mysql,andphp.2)uscodeeditorforsyntaxssupport.3)

如何通過php中的索引訪問字符串中的字符 如何通過php中的索引訪問字符串中的字符 Jul 12, 2025 am 03:15 AM

在PHP中獲取字符串特定索引字符可用方括號或花括號,但推薦方括號;索引從0開始,超出范圍訪問返回空值,不可賦值;處理多字節(jié)字符需用mb_substr。例如:$str="hello";echo$str[0];輸出h;而中文等字符需用mb_substr($str,1,1)獲取正確結果;實際應用中循環(huán)訪問前應檢查字符串長度,動態(tài)字符串需驗證有效性,多語言項目建議統(tǒng)一使用多字節(jié)安全函數(shù)。

快速PHP安裝教程 快速PHP安裝教程 Jul 18, 2025 am 04:52 AM

ToinstallPHPquickly,useXAMPPonWindowsorHomebrewonmacOS.1.OnWindows,downloadandinstallXAMPP,selectcomponents,startApache,andplacefilesinhtdocs.2.Alternatively,manuallyinstallPHPfromphp.netandsetupaserverlikeApache.3.OnmacOS,installHomebrew,thenrun'bre

See all articles