面试官问“装饰器模式”,这样回答薪资多要 3000!

简介: 本文用“奶茶加料”生动讲解PHP装饰器模式:当继承导致类爆炸时,改用组合——像包洋葱一样层层装饰对象。通过接口、基础类、抽象装饰器和具体装饰器四步实现,动态叠加功能,零修改扩展代码,完美践行开闭原则。(239字)

最近在 Code Review 的时候,看到某同事写代码,那叫一个“继承满天飞”。为了给一个核心类增加不同的功能,他硬生生造出了十几个子类。看着那像族谱一样庞大的类继承树,我手里的咖啡瞬间就不香了。

很多初学者(甚至老手)在面对 “如何在不修改原有类代码的情况下,动态地给对象增加功能” 这个问题时,第一反应往往是继承。

但是,朋友们,继承是把双刃剑。当你需要排列组合各种功能时,继承就会引发“类爆炸”。

今天,咱们不用枯燥的理论,用一杯奶茶为例,带大家由浅入深,彻底搞懂在 PHP 中如何使用:装饰器模式 (Decorator Pattern)


1. 痛点:当需求开始“套娃”

假设我们在开发一个奶茶店的收银系统。
最开始,需求很简单:只卖原味奶茶,一杯 10 块钱。

你可能会写一个 MilkTea 类,里面有个 cost() 方法返回 10。

老板的需求总是猝不及防:

“我们要加料!加珍珠(+2元)、加椰果(+3元)、加布丁(+4元)...”

如果你用继承,你可能会写出:

  • PearlMilkTea (珍珠奶茶)
  • CoconutMilkTea (椰果奶茶)
  • PuddingMilkTea (布丁奶茶)

这时候,顾客说:“我要一杯加珍珠、加椰果、半糖、去冰的奶茶。”

完了,难道你要写一个 PearlAndCoconutAndHalfSugarAndNoIceMilkTea 类吗?
如果配料有 10 种,排列组合下来的子类数量简直是天文数字。这就是传说中的“类爆炸”。


2. 破局:像“洋葱”一样包裹

装饰器模式的核心思想,就是组合优于继承

想象一下,我们不再通过生产新的奶茶品种来满足需求,而是把奶茶当作一个核心。

  • 加珍珠?就是在奶茶外面包一层“珍珠装饰器”。
  • 再加椰果?就在刚才的整体外面再包一层“椰果装饰器”。

这一层层包起来,就像俄罗斯套娃,或者像洋葱。每一层装饰器都只关心加上自己的价格,然后把任务交给里面的那一层。


3. 实战:用 PHP 一步一步撸代码

Talk is cheap, show me the code. 我们用 PHP 8 的语法来优雅地实现它。

第一步:定义统一的接口 (Contract)

首先,无论是原始的奶茶,还是加了料的奶茶,它们在本质上都是“饮品”。我们需要一个接口来规范它们。

<?php

// 饮品接口
interface Beverage
{
   
    // 获取描述
    public function getDescription(): string;

    // 获取价格
    public function cost(): int;
}

第二步:实现最纯粹的原始对象 (Concrete Component)

这是我们的基底,一杯朴实无华的原味奶茶。

class SimpleMilkTea implements Beverage
{
   
    public function getDescription(): string
    {
   
        return "原味奶茶";
    }

    public function cost(): int
    {
   
        return 10; // 基础价 10 元
    }
}

第三步:打造装饰器基类 (Base Decorator)

这是模式的精髓。装饰器本身也是 Beverage,但它内部持有一个 Beverage 对象。
注意:这里使用 abstract 是为了让子类去实现具体的加料逻辑。

abstract class BeverageDecorator implements Beverage
{
   
    // PHP 8 构造函数属性提升,直接注入被装饰的对象
    public function __construct(
        protected Beverage $beverage
    ) {
   }

    // 默认行为:直接调用里面那层的方法
    public function getDescription(): string
    {
   
        return $this->beverage->getDescription();
    }

    public function cost(): int
    {
   
        return $this->beverage->cost();
    }
}

第四步:具体的加料装饰器 (Concrete Decorators)

现在,我们想加什么料,就写什么类,随写随用,互不干扰。

加珍珠(Pearl):

class PearlDecorator extends BeverageDecorator
{
   
    public function getDescription(): string
    {
   
        // 先获取里面的描述,再追加自己的描述
        return $this->beverage->getDescription() . " + 珍珠";
    }

    public function cost(): int
    {
   
        // 核心价格 + 珍珠的价格(2元)
        return $this->beverage->cost() + 2;
    }
}

加布丁(Pudding):

class PuddingDecorator extends BeverageDecorator
{
   
    public function getDescription(): string
    {
   
        return $this->beverage->getDescription() . " + 布丁";
    }

    public function cost(): int
    {
   
        return $this->beverage->cost() + 4; // 布丁贵一点
    }
}

4. 见证奇迹的时刻

代码写好了,我们看看在业务逻辑中怎么使用。你会发现这种写法极其灵活。

// 1. 点一杯原味奶茶
$myDrink = new SimpleMilkTea();
echo "刚开始: " . $myDrink->getDescription() . " 价格:" . $myDrink->cost() . "\n";

// 2. 顾客说要加珍珠
// 把原味奶茶塞进珍珠装饰器里
$myDrink = new PearlDecorator($myDrink);

// 3. 顾客又说要加布丁
// 把刚才加了珍珠的奶茶,再塞进布丁装饰器里
$myDrink = new PuddingDecorator($myDrink);

echo "最终成品: " . $myDrink->getDescription() . "\n";
echo "最终价格: " . $myDrink->cost() . " 元\n";

运行结果:

刚开始: 原味奶茶 价格:10
最终成品: 原味奶茶 + 珍珠 + 布丁
最终价格: 16 元

看到了吗?我们可以无限地 new Decorator(new Decorator(...)) 套下去,完全不需要修改 SimpleMilkTea 的代码,也不需要创建复杂的继承树。


5. 什么时候使用?

不要手里拿着锤子,看什么都是钉子。只有在以下场景,强烈建议使用装饰器模式:

  1. 动态增强功能: 你需要在运行时给一个对象增加额外的职责(如:给文本加粗、给 HTTP 请求加 Token)。
  2. 避免继承爆炸: 当类变体过多,或者功能需要排列组合时。
  3. 遵循开闭原则 (OCP): 对扩展开放,对修改关闭。

在 PHP 著名的 Laravel 框架中,中间件 (Middleware) 的实现机制在本质上就是一种变种的装饰器模式(洋葱模型),请求穿过层层中间件,最后到达控制器。

恭喜你,现在知道了中间件的实现方式……

相关文章
|
4天前
|
人工智能 定位技术 SEO
我学 GEO 第 15 天:终于知道AI GEO该如何做?
我是暴走的莉莉酱,边旅行边研究AI GEO的数字游民。专注普通人如何提升“AI可见度”——让AI在回答用户问题时准确识别、理解并推荐你。不讲玄学,只做可测、可调、可持续的GEO实践。
394 124
|
6天前
|
机器学习/深度学习 人工智能 调度
🐴 HappyHorse 1.1 现已上线阿里云百炼!快来查收模型使用指南,现在调用享 6 折~
HappyHorse 1.1 是新一代视频生成大模型,全面升级动态表现力、角色一致性、指令遵循、视觉质感与音画协同能力。支持I2V/T2V/R2V三类生成,适配短剧、电商广告、品牌营销等场景,提供高质、流畅、可控的AI视频生产力。
669 4
🐴 HappyHorse 1.1 现已上线阿里云百炼!快来查收模型使用指南,现在调用享 6 折~
|
4天前
|
缓存 人工智能 运维
阿里云618百炼大模型Qwen3.7-Max功能、免费试用、订阅计费、配置接入详解
Qwen3.7-MAX是阿里云百炼平台推出的通义千问3.7系列旗舰大语言模型,专为智能体时代复杂任务打造,依托阿里云全域算力与自研技术,在逻辑推理、长文本处理、代码工程、长周期自主执行等领域达到行业顶尖水平。2026年618期间,该模型推出多重免费试用权益、按量计费5折、订阅套餐优惠等专属福利,覆盖个人开发者、团队与企业全场景需求,以下从核心功能、免费试用、订阅计费、配置接入四方面展开详细解析。
389 123
|
2天前
|
人工智能 自然语言处理 API
阿里云Token Plan团队版解析:功能、三档套餐与省钱订阅指南
阿里云百炼平台推出的Token Plan团队版,是面向企业与团队的AI大模型订阅服务,以Credits为统一计量单位,整合文本与图像生成模型,提供团队管理、数据安全、多工具兼容等核心能力,解决团队零散订阅AI服务的管理混乱、成本失控、数据安全等痛点。本文将从核心定位、套餐详情、计费规则、团队管理、工具兼容、便宜订阅技巧等方面,全面解析Token Plan团队版,帮助企业与团队高效、低成本地使用AI服务。
294 108
|
17天前
|
缓存 测试技术 API
Qwen 3.7 Plus 与 Max 实测:性价比与多模态能力差异解析(2026)
2026 年 6 月 1 日,阿里悄无声息地发布了 Qwen 3.7 Plus,距 Qwen 3.7 Max 上线刚好 11 天。同样的 1M 上下文,同样的 35 小时自治上限。但价格才是头条:Plus 是 0.40/M输入,Max是 2.50/M——便宜约 6 倍——并且还能看图、看视频。Vision Arena 上 Plus 已经排到 #16。所以这周真正值得讨论的问题不是”要不要为视觉能力买单”,而是”Max 凭什么用 6 倍价格换来 2 个百分点的 benchmark 领先”。
|
3天前
|
存储 人工智能 数据可视化
别再手动复制 Skill 了:多 Agent 时代的 Skill 管理方案
多 Agent 场景下 Skill 的统一管理与同步。
219 124
|
10天前
|
缓存 人工智能 运维
GLM 5.2自托管全流程实战:硬件选型、vLLM/SGLang部署与成本盈亏测算
2026年智谱发布GLM 5.2超大混合专家模型,区别于以往仅开放API的闭源大模型,该模型权重以MIT开源协议对外发布,企业与开发者可完整下载、本地审计、私有化部署,实现数据不出环境、自定义微调、自主调度推理资源。GLM 5.2拥有753B总参数,原生支持百万级上下文窗口,在代码生成、长文档推理、数学逻辑等多项基准测试中对标国际顶尖商用模型,是首款可完整自托管的前沿代码向大模型。
831 0
|
3天前
|
SQL 存储 运维
日志能不能改?SLS LogStore 原生支持更新和删除了
随着日志承载的业务语义越来越多,数据订正、回填、清理等需求变得越来越常见。SLS 现已为 LogStore 提供原生 update/delete 能力——支持按 RowID 精确修改,按查询条件批量操作,类似计费调账、标签刷新、反馈回填等场景都可以直接在 LogStore 内完成闭环。
188 124