设计模式——装饰器设计模式(结构型)

news2025/6/3 0:52:49

摘要

文中主要介绍了装饰器设计模式,它是一种结构型设计模式,可在不改变原有类代码的情况下,动态为对象添加额外功能。文中详细阐述了装饰器模式的角色、结构、实现方式、适合场景以及实战示例等内容,还探讨了其与其他设计模式的结合使用,帮助读者全面理解装饰器设计模式。

1. 装饰器设计模式定义

装饰器模式是一种结构型设计模式,它允许在不改变原有类代码的前提下,动态地为对象添加额外的功能。

  • 装饰器就像**“套娃”“穿衣服”**,一层一层包裹原始对象,增强它的功能。
  • 被装饰的对象不知道自己被增强了,增强逻辑是外部“包”实现的。

角色说明:

角色

含义

Component(抽象构件)

原始功能的抽象定义,通常是一个接口或抽象类。

ConcreteComponent(具体构件)

原始功能的具体实现类。

Decorator(装饰器抽象类)

包装 Component,并定义扩展接口,通常持有一个 Component 引用。

ConcreteDecorator(具体装饰器)

扩展 Component 功能的类,实现增强逻辑。

2. 装饰器设计模式结构

2.1. 装饰器类图

装饰模式包含如下角色:

  • Component: 抽象构件
  • ConcreteComponent: 具体构件
  • Decorator: 抽象装饰类
  • ConcreteDecorator: 具体装饰类

2.2. 装饰器时序图

3. 装饰器设计模式实现方式

装饰器设计模式有三种常见实现方式,关键思想是一致的:将原对象“包裹”起来,扩展功能而不改变原代码

3.1. 常见实现方式

实现方式

描述

推荐度

接口 + 抽象类(经典做法)

定义接口 + 装饰器抽象类 + 具体装饰类

⭐⭐⭐⭐

仅使用接口或抽象类(无抽象装饰器)

简化结构,直接实现装饰逻辑

⭐⭐⭐

使用 Java 动态代理或 Spring AOP

用代理机制动态增强对象行为

⭐⭐⭐⭐⭐(现代项目常用)

3.2. 🧱 示例:给“原始服务”动态添加日志和鉴权功能

3.2.1. 🎯 目标接口(Component)

public interface OrderService {
    void placeOrder(String user);
}

3.2.2. ✅ 原始类(ConcreteComponent)

public class SimpleOrderService implements OrderService {
    @Override
    public void placeOrder(String user) {
        System.out.println("下单成功:" + user);
    }
}

3.3. ✅ 实现方式一:接口 + 抽象装饰器类(经典方式)

3.3.1. 🎯 抽象装饰器类(Decorator)

public abstract class OrderServiceDecorator implements OrderService {
    
    protected OrderService delegate;

    public OrderServiceDecorator(OrderService delegate) {
        this.delegate = delegate;
    }
}

3.3.2. 🧩 具体装饰器 1:日志功能

public class LoggingOrderService extends OrderServiceDecorator {
    
    public LoggingOrderService(OrderService delegate) {
        super(delegate);
    }

    @Override
    public void placeOrder(String user) {
        System.out.println("[日志] 用户:" + user + " 正在下单...");
        delegate.placeOrder(user);
    }
}

3.3.3. 🧩 具体装饰器 2:权限功能

public class AuthOrderService extends OrderServiceDecorator {
    
    public AuthOrderService(OrderService delegate) {
        super(delegate);
    }

    @Override
    public void placeOrder(String user) {
        if (!"admin".equals(user)) {
            throw new SecurityException("无权限");
        }
        delegate.placeOrder(user);
    }
}

3.3.4. ✅ 使用方式:

OrderService service = new SimpleOrderService();
service = new LoggingOrderService(service);
service = new AuthOrderService(service);

service.placeOrder("admin"); // ✅ 正常
// service.placeOrder("guest"); // ❌ 抛出无权限异常

3.4. ✅ 实现方式二:Spring AOP(自动代理装饰)

如果你用 Spring,可以用 AOP 注解来“无侵入式”地增强行为 —— 本质也是装饰器。

@Aspect
@Component
public class LogAspect {

    @Before("execution(* com.example.OrderService.placeOrder(..))")
    public void logBefore() {
        System.out.println("[AOP日志] 开始下单...");
    }
}

3.5. ✅ 实现方式三:Java 动态代理(JDK Proxy)

OrderService target = new SimpleOrderService();
OrderService proxy = (OrderService) Proxy.newProxyInstance(
    target.getClass().getClassLoader(),
    new Class[]{OrderService.class},
    (proxyObj, method, args) -> {
        System.out.println("[代理] 下单前检查");
        return method.invoke(target, args);
    }
);
proxy.placeOrder("admin");

3.6. ✅ 装饰器实现方式总结

实现方式

特点

场景推荐

抽象类 + 接口

结构清晰、经典实现

适合手动控制、多个装饰层

仅接口实现

简洁灵活

适合小项目或简单增强

动态代理 / Spring AOP

自动增强、无侵入

Spring 项目首选

4. 装饰器设计模式适合场景

4.1. ✅ 适合使用装饰器模式的场景

场景

说明

需要扩展对象功能且不影响原类代码

不修改原类,动态增加功能,如日志、鉴权、限流。

需要在运行时动态组合多种行为

例如对同一个服务动态添加多种增强层:缓存、监控、安全等。

功能可以“嵌套组合”叠加

类似“加配料”的逻辑,如流式处理、文本格式处理、消息管道处理。

不能使用继承或继承组合会导致类爆炸

多个子类组合场景,如A+B+C,用装饰器组合比继承更灵活。

希望通过组合替代继承,增强可扩展性

符合“开闭原则”,增加功能时只加类、不改原有代码。

Spring 项目中常见增强逻辑(AOP 本质)

权限验证、日志记录、事务管理、指标统计等。

4.2. ❌ 不适合使用装饰器模式的场景

场景

原因

功能变化不是可组合的,而是互斥的

例如“单选”的策略或“开关”逻辑,不适合一层层叠加。

装饰层次太多导致调用链复杂、调试困难

层数一多,调用顺序和调试堆栈难以理解和排查。

功能变化不明显,使用装饰器反而增加复杂度

只是简单逻辑改造,不值得为了“设计模式”增加结构。

需要共享状态或强耦合多个功能模块

装饰器是包装独立功能,若需要强耦合交互,不合适。

对性能极度敏感、低延迟要求场景

层层封装增加方法调用和对象创建成本,可能不划算。

4.3. 🧠 总结对比表格

项目

✅ 适合使用

❌ 不适合使用

是否需要动态增强

✅ 是

❌ 否

是否支持组合功能

✅ 是

❌ 否,功能互斥

原类是否可修改

❌ 否(不能改)

✅ 是(能改就没必要)

是否需要多个扩展行为

✅ 是

❌ 否(只需单一变化)

是否可读性、调试性良好

✅ 层数少

❌ 层数多导致复杂

是否存在替代方案

❌ 无法用继承或策略代替

✅ 策略、简单组合更合适

5. 装饰器设计模式实战示例

在风控系统中,装饰器设计模式非常适合用来扩展规则校验、风控拦截、日志、监控、异常报警等横切功能,而且它不改变原有核心风控逻辑,具备很强的可组合性和可扩展性。

5.1. ✅ 业务场景:用户申请贷款,风控系统对其进行多层校验

5.2. 🎯 目标:

风控规则如下,可独立或组合使用:

  1. 黑名单校验
  2. 年龄校验
  3. 欺诈用户识别
  4. 日志记录

5.3. 🧱 定义风控接口(Component)

public interface RiskCheck {
    boolean check(RiskContext context);
}

5.4. 🧱 定义风控上下文(数据输入)

@Data
public class RiskContext {
    private String userId;
    private int age;
    private boolean inBlacklist;
    private boolean isFraud;

    // 构造、getter/setter 省略
}

5.5. 🧱 基础校验类(ConcreteComponent)

public class BasicRiskCheck implements RiskCheck {
    @Override
    public boolean check(RiskContext context) {
        return true; // 默认无拦截
    }
}

5.6. 🧱 装饰器抽象类(Decorator)

public abstract class RiskCheckDecorator implements RiskCheck {
    
    protected RiskCheck delegate;

    public RiskCheckDecorator(RiskCheck delegate) {
        this.delegate = delegate;
    }
}

5.7. ✅ 具体装饰器实现(ConcreteDecorator)

5.7.1. 黑名单校验:

public class BlacklistCheck extends RiskCheckDecorator {
    
    public BlacklistCheck(RiskCheck delegate) {
        super(delegate);
    }

    @Override
    public boolean check(RiskContext context) {
        if (context.isInBlacklist()) {
            System.out.println("拦截:用户在黑名单中");
            return false;
        }
        return delegate.check(context);
    }
}

5.7.2. 年龄校验:

public class AgeCheck extends RiskCheckDecorator {
    
    public AgeCheck(RiskCheck delegate) {
        super(delegate);
    }

    @Override
    public boolean check(RiskContext context) {
        if (context.getAge() < 18) {
            System.out.println("拦截:未成年人禁止贷款");
            return false;
        }
        return delegate.check(context);
    }
}

5.7.3. 欺诈识别:

public class FraudCheck extends RiskCheckDecorator {
    
    public FraudCheck(RiskCheck delegate) {
        super(delegate);
    }

    @Override
    public boolean check(RiskContext context) {
        if (context.isFraud()) {
            System.out.println("拦截:疑似欺诈用户");
            return false;
        }
        return delegate.check(context);
    }
}

5.7.4. 日志记录:

public class LoggingCheck extends RiskCheckDecorator {
    
    public LoggingCheck(RiskCheck delegate) {
        super(delegate);
    }

    @Override
    public boolean check(RiskContext context) {
        System.out.println("执行风控检查...");
        boolean result = delegate.check(context);
        System.out.println("风控结果:" + result);
        return result;
    }
}

5.8. ✅ 组装装饰链(动态组合)

RiskCheck riskCheck = new BasicRiskCheck();
riskCheck = new LoggingCheck(riskCheck);
riskCheck = new BlacklistCheck(riskCheck);
riskCheck = new AgeCheck(riskCheck);
riskCheck = new FraudCheck(riskCheck);

5.9. 🧪 调用示例

RiskContext context = new RiskContext();
context.setUserId("u123");
context.setAge(20);
context.setInBlacklist(false);
context.setFraud(false);

boolean result = riskCheck.check(context);
System.out.println("最终是否通过风控:" + result);

5.10. 🧠 总结优势

优点

描述

动态组合风控规则

根据不同产品或渠道,组合不同装饰器

遵守开闭原则

添加新风控规则无需改动原有类,只需新增装饰器类

可重用、可插拔

每个装饰器是一个独立功能模块,支持复用

与 Spring 兼容性好

可以用 @Component + @Conditional 结合使用

5.11. 📦 Bonus:可结合策略模式 & 装饰器做更强动态配置

比如:

  • 风控策略由数据库或配置中心定义
  • 然后动态组合装饰器链
    是否也想看这个高级版本?我可以给你代码。

6. 装饰器设计模式思考

博文参考

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2394320.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

途景VR智拍APP:开启沉浸式VR拍摄体验

在数字化时代&#xff0c;VR技术以其沉浸式的体验逐渐走进了人们的日常生活。途景VR智拍APP作为一款集看图和拍照于一体的VR软件&#xff0c;为用户带来了全新的视觉体验和便捷的拍摄方式&#xff0c;无论是专业摄影师还是普通用户&#xff0c;都能轻松上手&#xff0c;拍出令人…

Linux环境搭建MCU开发环境

操作系统版本&#xff1a; ubuntu 22.04 文本编辑器&#xff1a; vscode 开发板&#xff1a; stm32f103c8t6 调试器&#xff1a; st-link 前言 步骤一&#xff1a; 安装交叉编译工具链 步骤二&#xff1a; 创建工程目录结构 步骤三&#xff1a; 调试…

【基础算法】高精度(加、减、乘、除)

文章目录 什么是高精度1. 高精度加法解题思路代码实现 2. 高精度减法解题思路代码实现 3. 高精度乘法解题思路代码实现 4. 高精度除法 (高精度 / 低精度)解题思路代码实现 什么是高精度 我们平时使用加减乘除的时候都是直接使用 - * / 这些符号&#xff0c;前提是进行运算的数…

Windows最快速打开各项系统设置大全

目录 一、应用背景 二、设置项打开方法 2.1 方法一界面查找&#xff08;最慢&#xff09; 2.2 方法二cmd命令&#xff08;慢&#xff09; 2.3 方法三快捷键&#xff08;快&#xff09; 2.4 方法四搜索栏&#xff08;快&#xff09; 2.5 方法五任务栏&#xff08;最快&am…

嵌入式编译工具链熟悉与游戏移植

在自己的虚拟机Ubuntu系统下&#xff0c;逐步编译 mininim源码(波斯王子重制开源版&#xff09; 指令流程 sudo apt-get remove liballegro5-dev liballegro-image5-dev \liballegro-audio5-dev liballegro-acodec5-dev liballegro-dialog5-dev sudo apt-get install automak…

DeepSeek-R1-0528,官方的端午节特别献礼

DeepSeek&#xff1a;端午安康&#xff01;刻在国人骨子里的浪漫 2025 年 05 月 28 日 | DeepSeek 端午特别献礼 当粽叶飘香时&#xff0c;DeepSeek 悄然带来一份节日惊喜 版本号 DeepSeek-R1-0528 正式上线 官方赋予它的灵魂是&#xff1a; 思考更深 推理更强 用户通过官网…

001 flutter学习的注意事项及前期准备

在学习flutter之前&#xff0c;还需要进行一些初始的配置&#xff0c;然后才可以学习flutter 1.安装flutter 国内官网&#xff1a;https://flutter.cn​​​​​​ 国际官网&#xff1a;https://flutter.dev 安装完成后&#xff0c;按照官网上面的操作步骤进行配置&#xf…

CS144 - Lecture 1 记录

CS144 - Lecture 1 由于没讲义&#xff0c;全看课了&#xff0c;系统性的总结有点难&#xff0c;记一些有趣的东西吧。 数据链路和网络层的传输 我们可以看见&#xff0c;对于发送方&#xff0c;我们的数据链路层为我们的网络层提供服务&#xff0c;在经过路由的时候&#xf…

【数据结构】——二叉树--链式结构

一、实现链式结构二叉树 二叉树的链式结构&#xff0c;那么从名字上我们就知道我们这个二叉树的底层是使用链表来实现的&#xff0c;前面我们的二叉树是通过数组来实现的&#xff0c;那么在其是完全二叉树的情况下&#xff0c;此时我们使用数组来实现就会使得其空间浪费较少&a…

充电便捷,新能源汽车移动充电服务如何预约充电

随着新能源汽车的普及&#xff0c;充电便捷性成为影响用户体验的关键因素之一。传统的固定充电桩受限于地理位置和数量&#xff0c;难以完全满足用户需求&#xff0c;而移动充电服务的出现&#xff0c;为车主提供了更加灵活的补能方式。通过手机APP、小程序或在线平台&#xff…

基于 Chrome 浏览器扩展的Chroma简易图形化界面

简介 ChromaDB Manager 是基于 Chrome 浏览器扩展的一款 ChromaDB&#xff08;一个流行的向量数据库&#xff09;的数据查询工具。提供了一个用户友好的界面&#xff0c;可以直接从浏览器连接到本地 ChromaDB 实例、查看集合信息和分片数据。本工具特别适合开发人员快速查看和…

IM系统的负载均衡

1.IM场景的负载均衡 2.方案总览 SDK层想要连接一个TCP网关或者WebSocket网关的方案 SDK单地址:在SDK中写死某个网关的IP或者域名,缺点是更换地址需要重新打包SDK SDK多地址:防止某一个地址嗝屁了写上多个地址用足保持高可用 暴露接口给客户端:SDK层访问接口动态获得地址 注…

使用 Zabbix 监控 MySQL 存储空间和性能指标的完整实践指南

目录 引言 一、最终目标支持功能 二、监控方案设计 2.1 技术选型 2.2 设计思路 三、实现步骤 3.1 准备工作 3.11 创建 MySQL 监控账号 3.12 配置 .my.cnf 文件 3.2 编写统一脚本 3.3 配置 Zabbix Agent UserParameter 3.4 Zabbix 前端配置建议 四、总结 引言 MySQL …

【技能拾遗】——家庭宽带单线复用布线与配置(移动2025版)

&#x1f4d6; 前言&#xff1a;在家庭网络拓扑中&#xff0c;客厅到弱电箱只预埋了一根网线&#xff0c;由于已将广电的有线电视取消并改用IPTV。现在需要解决在客厅布置路由器和观看IPTV问题&#xff0c;这里就用到单线复用技术。 目录 &#x1f552; 1. 拓扑规划&#x1f55…

异步日志监控:FastAPI与MongoDB的高效整合之道

title: 异步日志监控:FastAPI与MongoDB的高效整合之道 date: 2025/05/27 17:49:39 updated: 2025/05/27 17:49:39 author: cmdragon excerpt: FastAPI与MongoDB整合实现日志监控系统的实战指南。首先配置MongoDB异步连接,定义日志数据模型。核心功能包括日志写入接口、聚合…

在 Android 上备份短信:保护您的对话

尽管我们的Android手机有足够的存储空间来存储无数的短信&#xff0c;但由于设备故障、意外删除或其他意外原因&#xff0c;您可能会丢失重要的对话。幸运的是&#xff0c;我们找到了 5 种有效的 Android SMS 备份解决方案&#xff0c;确保您的数字聊天和信息保持安全且可访问。…

标题:2025海外短剧爆发年:APP+H5双端系统开发,解锁全球流量与变现新大陆

描述&#xff1a; 2025年出海新风口&#xff01;深度解析海外短剧系统开发核心&#xff08;APPH5双端&#xff09;&#xff0c;揭秘高效开发策略与商业化路径&#xff0c;助您抢占万亿美元市场&#xff01; 全球娱乐消费模式正在剧变。2025年&#xff0c;海外短剧市场已从蓝海…

解决RAGFlow(v0.19.0)有部分PDF无法解析成功的问题。

ragflow版本为&#xff1a;v0.19.0 1.解析的时候报错&#xff1a;Internal server error while chunking: Coordinate lower is less than upper。 看报错怀疑是分片的问题&#xff0c;于是把文档的切片方法中的“建议文本块大小”数值&#xff08;默认512&#xff09;调小&…

c#基础08(数组)

文章目录 数组数组概念声明数组初始化数组赋值给数组访问数组元素 集合动态数组(ArrayList)使用foreach循环C#数组细节多维数组传递数组给函数参数数组 数组 数组概念 数组是一个存储相同类型元素的固定大小的顺序集合。数组是用来存储数据的集合&#xff0c;通常认为数组是一…

嵌入式学习--江协stm32day3

这是我目前为止认为最重要的模块--TIM定时器&#xff0c;这里我们主要学习通用定时器 最小的计数计时单元为时基单元&#xff0c;包括PSC&#xff0c;ARR&#xff0c;CNT CK_PSC&#xff08;Prescaler&#xff0c;预分频器&#xff09;&#xff1a;作用是对输入时钟信号进行分…