橋接模式,將實作體系與抽象體系分離開來,讓兩者能各自更動各自演進,就有點像是大頭菜有分健康的大頭菜及壞掉的大頭菜,你的島上有這兩種大頭菜,但是健康的大頭菜過了一個禮拜都沒賣掉的話,他就會變成壞掉的大頭菜了。
什麼是橋接模式#
核心概念#
橋接模式核心概念
橋接模式將抽象部分與實作部分分離,使它們都可以獨立地變化。這裡的分離是指在抽象和實作之間使用組合/聚合關係,而不是繼承關係,從而使兩者可以相對獨立地變化。
使用時機#
適用情境
- 不希望在抽象和實作部分之間有固定的繫結關係
- 抽象部分和實作部分都可能需要擴展
- 希望對客戶隱藏實作細節
- 需要在多個物件間共享實作
- 想要實現「組合優於繼承」的設計原則
UML#

實作#
建立實作部分(Implementation)#
實作步驟
首先我們要來把大頭菜做出來,因此會有定義大頭菜介面 Interface、建立健康的大頭菜及壞掉的大頭菜。這些類別組成了橋接模式中的「實作部分」。
TurnipsInterface.php#
/**
 * Interface TurnipsInterface.
 */
interface TurnipsInterface
{
    public function calculatePrice(): int;
}
Turnips.php#
/**
 * Class Turnips.
 */
class Turnips implements TurnipsInterface
{
    /**
     * @var int
     */
    protected $price;
    /**
     * @var int
     */
    protected $count;
    /**
     * Turnips constructor.
     *
     * @param int $price
     * @param int $count
     */
    public function __construct(int $price, int $count)
    {
        $this->price = $price;
        $this->count = $count;
    }
    /**
     * @return int
     */
    public function calculatePrice(): int
    {
        if (isset($this->price) && isset($this->count)) {
            return $this->price * $this->count;
        } else {
            return 0;
        }
    }
}
Spoiled.php#
/**
 * Class Spoiled.
 */
class Spoiled implements TurnipsInterface
{
    /**
     * @var int
     */
    protected $price;
    /**
     * @var int
     */
    protected $count;
    /**
     * Spoiled constructor.
     *
     * @param int $price
     * @param int $count
     */
    public function __construct(int $price, int $count)
    {
        $this->price = $price;
        $this->count = $count;
    }
    /**
     * @return int
     */
    public function calculatePrice(): int
    {
        if (isset($this->price) && isset($this->count)) {
            return 0 * $this->count;
        } else {
            return 0;
        }
    }
}
建立抽象部分(Abstraction)#
橋樑設計重點
接下來我們要建立所謂的橋樑,這裡有個重要的地方是實作(Implementation),功能是來去替換大頭菜的類別。這個部分組成了橋接模式中的「抽象部分」,它包含了對實作部分的參照。
BaseService.php#
/**
 * Class BaseService.
 */
abstract class BaseService
{
    /**
     * @var TurnipsInterface
     */
    protected $implementation;
    /**
     * @param TurnipsInterface $turnips
     */
    public function __construct(TurnipsInterface $turnips)
    {
        $this->implementation = $turnips;
    }
    /**
     * @param TurnipsInterface $turnips
     */
    public function setImplementation(TurnipsInterface $turnips)
    {
        $this->implementation = $turnips;
    }
    abstract public function calculatePrice();
}
TurnipsService.php#
/**
 * Class TurnipsService.
 */
class TurnipsService extends BaseService
{
    /**
     * @return int
     */
    public function calculatePrice()
    {
        return $this->implementation->calculatePrice();
    }
}
測試#
測試目標#
最後我們要來寫幾些測試項目,主要測試建立 Service 並且執行計算,以及是否能夠替換大頭菜類別,並且計算。
測試清單
- 健康大頭菜測試: 測試是否能建立 Service 並把健康的大頭菜塞進去,然後計算鈴錢 
- 壞掉大頭菜測試: 測試是否能建立 Service 並把壞掉的大頭菜塞進去,然後計算鈴錢 
- 橋接更換測試: 測試是否能建立 Service 並把健康的大頭菜塞進去,然後計算鈴錢,再把大頭菜替換成壞掉的大頭菜,再計算一次鈴錢 
測試程式碼#
BridgePatternTest.php#
/**
 * Class BridgePatternTest.
 */
class BridgePatternTest extends TestCase
{
    /**
     * 測試是否能建立 Service 並把健康的大頭菜塞進去,然後計算鈴錢。
     * 
     * @test
     */
    public function test_can_calculate_price_for_turnips()
    {
        $service = new TurnipsService(new Turnips(100, 40));
        $this->assertEquals(4000, $service->calculatePrice());
    }
    
    /**
     * 測試是否能建立 Service 並把壞掉的大頭菜塞進去,然後計算鈴錢。
     * 
     * @test
     */
    public function test_can_calculate_price_for_spoiled()
    {
        $service = new TurnipsService(new Spoiled(100, 40));
        $this->assertEquals(0, $service->calculatePrice());
    }
    
    /**
     * 測試是否能建立 Service 並把健康的大頭菜塞進去,然後計算鈴錢,
     * 再把大頭菜替換成壞掉的大頭菜,再計算一次鈴錢。
     * 
     * @test
     */
    public function test_can_calculate_price_for_turnips_and_spoiled()
    {
        $service = new TurnipsService(new Turnips(100, 40));
        $this->assertEquals(4000, $service->calculatePrice());
        $service->setImplementation(new Spoiled(100, 40));
        $this->assertEquals(0, $service->calculatePrice());
    }
}
測試結果#
最後測試的執行結果會獲得如下:
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.
 ==> AbstractFactoryTest        ✔  ✔  ✔  ✔  
 ==> BuilderPatternTest         ✔  ✔  ✔  ✔  
 ==> FactoryMethodTest          ✔  ✔  ✔  ✔  
 ==> PoolPatternTest            ✔  ✔  
 ==> PrototypePatternTest       ✔  ✔  
 ==> SimpleFactoryTest          ✔  ✔  ✔  ✔  
 ==> SingletonPatternTest       ✔  
 ==> StaticFactoryTest          ✔  ✔  ✔  ✔  ✔  
 ==> AdapterPatternTest         ✔  ✔  
 ==> BridgePatternTest          ✔  ✔  ✔  
Time: 00:00.027, Memory: 6.00 MB
OK (31 tests, 76 assertions)

