中介者模式,在兩個不同的封裝物件之間,作為中間進行交互的模式,可以減少物件之間的依賴關係,並且降低耦合性問題,舉例來說有背包(Bag)與商店(Store)這兩個物件,你會從背包(Bag)當中拿出鈴錢(Bells)去商店(Store)購買大頭菜(Turnips),但它們應該要各自其職,不要太過於互相依賴,因此你會需要有個中間控制這些物件的中介者(Mediator)。
UML
實作 首先我們不外乎先製作出大頭菜(Turnips)物件,只需要賦予簡單計算數量的功能即可。
Turnips.php
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 class Turnips { protected int $count ; public function __construct (int $count ) { $this ->setCount ($count ); } public function setCount (int $count ) { $this ->count = $count ; } public function getCount ( ): int { return $this ->count; } }
接下來我們要做出背包(Bag)以及商店(Store)這兩個物件,因為這兩個物件會進行交互,所以需要有一個抽象類別去把中介者(Mediator)帶入,這裡稱之為協調同事(Colleague)物件,所以我們要先實作出協調同事(Colleague)這項物件,再去實作背包(Bag)以及商店(Store)這兩個物件,並且繼承協調同事(Colleague)這個抽象。
Colleague.php
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 abstract class Colleague { protected Mediator $mediator ; public function setMediator (Mediator $mediator ) { $this ->mediator = $mediator ; } }
Bag.php
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 class Bag extends Colleague { protected $bells ; protected $turnips ; public function __construct ( ) { $this ->bells = new Bells (0 ); $this ->turnips = new Turnips (0 ); } public function getBells ( ): int { return $this ->bells->getBells (); } public function getTurnips ( ): int { return $this ->turnips->getCount (); } public function setBells (int $bells ) { echo "[背包] 剩下 $bells 鈴錢。" ; $this ->bells->setBells ($bells ); } public function setTurnips (int $count ) { echo "[背包] 剩下 $count 顆大頭菜。" ; $this ->turnips->setCount ($count ); } }
Store.php
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 use InvalidArgumentException ;class Store extends Colleague { public function buyTurnips (int $price , int $count ) { $total = $price * $count ; if ($this ->mediator->getBells () >= $total ) { $this ->mediator->setBells ($this ->mediator->getBells () - $total ); echo "[商店] 收到了 $total 鈴錢。" ; echo "[商店] 賣出了 $count 顆大頭菜。" ; $this ->mediator->setTurnips ($this ->mediator->getTurnips () + $count ); return ; } throw new InvalidArgumentException ('[錯誤] 您背包裡的鈴錢不足,無法購買大頭菜。' ); } public function sellTurnips (int $price , int $count ) { if ($this ->mediator->getTurnips () >= $count ) { $this ->mediator->setTurnips ($this ->mediator->getTurnips () - $count ); echo "[商店] 收購了 $count 顆大頭菜。" ; $total = $price * $count ; echo "[商店] 支出了 $total 鈴錢。" ; $this ->mediator->setBells ($this ->mediator->getBells () + $total ); return ; } throw new InvalidArgumentException ('[錯誤] 您背包裡的大頭菜不足,無法販賣大頭菜。' ); } }
最後我們要來把中介者(Mediator)實作出來,這個中介者(Mediator)會作為背包(Bag)以及商店(Store)之間交互的載體,所以在實作之前先定義它的介面(Interface)出來。
MediatorInterface.php
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 interface MediatorInterface { public function getBells ( ): int ; public function getTurnips ( ): int ; public function setBells (int $bells ) ; public function setTurnips (int $count ) ; public function buyTurnips (int $price , int $count ) ; public function sellTurnips (int $price , int $count ) ; }
最後去實作出背包(Bag)與商店(Store)的中介者(Mediator),背包(Bag)主要控制鈴錢(Bells)以及大頭菜(Turnips),商店(Store)則是會透過中介者(Mediator)去向背包(Bag)去索取鈴錢(Bells)以及大頭菜(Turnips),而不是真的去觸及背包(Bag),減少物件之間的依賴關係。
BagStoreMediator.php
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 54 55 56 57 58 59 60 61 62 63 64 65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93 94 95 96 97 98 99 100 101 102 use InvalidArgumentException ;class BagStoreMediator implements MediatorInterface { protected Bag $bag ; protected Store $store ; public function __construct (Bag $bag , Store $store ) { $this ->bag = $bag ; $this ->store = $store ; $this ->bag->setMediator ($this ); $this ->store->setMediator ($this ); } public function getBells ( ): int { return $this ->bag->getBells (); } public function getTurnips ( ): int { return $this ->bag->getTurnips (); } public function setBells (int $bells ) { $this ->bag->setBells ($bells ); } public function setTurnips (int $count ) { $this ->bag->setTurnips ($count ); } public function buyTurnips (int $price , int $count ) { $total = $price * $count ; if ($this ->bag->getBells () >= $total ) { echo "[玩家] 您購買了 $count 顆大頭菜,每顆單價 $price 鈴錢,總共 $total 鈴錢。" ; $this ->store->buyTurnips ($price , $count ); return ; } throw new InvalidArgumentException ('[錯誤] 您的大頭菜不足,無法購買大頭菜。' ); } public function sellTurnips (int $price , int $count ) { $total = $price * $count ; if ($this ->bag->getTurnips () >= $count ) { echo "[玩家] 您販賣了 $count 顆大頭菜,每顆單價 $price 鈴錢,總共 $total 鈴錢。" ; $this ->store->sellTurnips ($price , $count ); return ; } throw new InvalidArgumentException ('[錯誤] 您的大頭菜不足,無法販賣大頭菜。' ); } }
測試 最後根據上面的中介者,我們要做幾個簡單的測試,第一件事是測試進行買賣大頭菜時,所產生的 Log 順序、資料是否正確,其次是例外錯誤 InvalidArgumentException 是否有確實。
MediatorPatternTest.php
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50 51 52 53 class MediatorPatternTest extends TestCase { public function test_buy_and_sell_turnips ( ) { $mediator = new BagStoreMediator (new Bag (), new Store ()); $mediator ->setBells (10000 ); $this ->expectOutputString (implode (array ( "[背包] 剩下 10000 鈴錢。" , "[玩家] 您購買了 40 顆大頭菜,每顆單價 100 鈴錢,總共 4000 鈴錢。" , "[背包] 剩下 6000 鈴錢。" , "[商店] 收到了 4000 鈴錢。" , "[商店] 賣出了 40 顆大頭菜。" , "[背包] 剩下 40 顆大頭菜。" , "[玩家] 您販賣了 20 顆大頭菜,每顆單價 200 鈴錢,總共 4000 鈴錢。" , "[背包] 剩下 20 顆大頭菜。" , "[商店] 收購了 20 顆大頭菜。" , "[商店] 支出了 4000 鈴錢。" , "[背包] 剩下 10000 鈴錢。" , ))); $mediator ->buyTurnips (100 , 40 ); $mediator ->sellTurnips (200 , 20 ); } public function test_buy_turnips_overflow ( ) { $this ->expectException (InvalidArgumentException ::class ); $mediator = new BagStoreMediator (new Bag (), new Store ()); $mediator ->buyTurnips (100 , 200 ); } public function test_sell_turnips_overflow ( ) { $this ->expectException (InvalidArgumentException ::class ); $mediator = new BagStoreMediator (new Bag (), new Store ()); $mediator ->sellTurnips (100 , 40 ); } }
最後測試的執行結果會獲得如下:
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30 31 32 33 PHPUnit Pretty Result Printer 0.28.0 by Codedungeon and contributors. ==> Configuration: ~/php-design-pattern/vendor/codedungeon/phpunit-result-printer/src/phpunit-printer.yml PHPUnit 9.2.6 by Sebastian Bergmann and contributors. ==> ...fResponsibilitiesTest ✔ ✔ ✔ ==> CommandPatternTest ✔ ==> IteratorPatternTest ✔ ✔ ✔ ✔ ==> MediatorPatternTest ✔ ✔ ✔ ==> AbstractFactoryTest ✔ ✔ ✔ ✔ ==> BuilderPatternTest ✔ ✔ ✔ ✔ ==> FactoryMethodTest ✔ ✔ ✔ ✔ ==> PoolPatternTest ✔ ✔ ==> PrototypePatternTest ✔ ✔ ==> SimpleFactoryTest ✔ ✔ ✔ ✔ ==> SingletonPatternTest ✔ ==> StaticFactoryTest ✔ ✔ ✔ ✔ ✔ ==> AdapterPatternTest ✔ ✔ ==> BridgePatternTest ✔ ✔ ✔ ==> CompositePatternTest ✔ ✔ ✔ ==> DataMapperTest ✔ ✔ ==> DecoratorPatternTest ✔ ✔ ==> DependencyInjectionTest ✔ ✔ ✔ ==> FacadePatternTest ✔ ==> FluentInterfaceTest ✔ ==> FlyweightPatternTest ✔ ==> ProxyPatternTest ✔ ✔ ==> RegistryPatternTest ✔ ✔ ✔ ✔ ✔ Time: 00:00.173, Memory: 6.00 MB OK (62 tests, 129 assertions)
完整程式碼 設計模式不難,找回快樂而已,以大頭菜為例。
參考文獻