【设计模式】深入解析装饰器模式(Decorator Pattern)
深入解析装饰器模式(Decorator Pattern)
一、装饰器模式的核心概念
装饰器模式是一种结构型设计模式,用于动态地给对象添加新功能,而不改变其原始代码。
1. 为什么需要装饰器?
- 避免继承带来的类爆炸问题:如果每种新功能都创建子类,组合复杂时类会爆炸式增长。
- 支持动态扩展功能:继承是静态的,而装饰器可以在运行时动态添加/移除功能。
- 遵循开闭原则(OCP):不修改原始代码,而是通过装饰器增强。
二、装饰器的核心机制
装饰器本质上就是一个高阶函数,它接收一个函数(或类)作为参数,返回一个新的函数(或类),增强原有功能。
装饰器的执行顺序
- 定义装饰器
- 编写一个函数,接收原函数
func
,在wrapper
中增加新功能。
- 编写一个函数,接收原函数
- 应用装饰器
- 通过
@decorator
语法,把装饰器作用在目标函数上。
- 通过
- 调用目标函数
- 实际执行的是
wrapper()
,它先执行装饰逻辑,再调用原函数。
- 实际执行的是
三、Python 装饰器示例
1. 计时装饰器
📌 示例:在函数运行前后打印执行时间
import time# 定义装饰器
def time_logger(func):def wrapper(*args, **kwargs):start_time = time.time() # 记录开始时间result = func(*args, **kwargs) # 执行原函数end_time = time.time() # 记录结束时间print(f"{func.__name__} 执行时间: {end_time - start_time:.6f} 秒")return result # 返回原函数的返回值return wrapper # 返回包装后的函数# 使用装饰器
@time_logger
def slow_function():time.sleep(1)print("函数执行完毕")slow_function()
执行流程:
@time_logger
作用在slow_function
上,相当于slow_function = time_logger(slow_function)
。- 当调用
slow_function()
时,实际上执行的是wrapper()
:- 先 记录时间
start_time
。 - 再 调用
slow_function()
。 - 最后 计算并打印执行时间。
- 先 记录时间
输出示例:
函数执行完毕
slow_function 执行时间: 1.000123 秒
2. 日志装饰器
📌 示例:在函数执行前后自动记录日志
def log_decorator(func):def wrapper(*args, **kwargs):print(f"开始执行 {func.__name__},参数:{args} {kwargs}")result = func(*args, **kwargs)print(f"{func.__name__} 执行完毕,返回值:{result}")return resultreturn wrapper@log_decorator
def add(a, b):return a + bprint(add(3, 5))
执行流程:
@log_decorator
把add
包装成wrapper
,add
变成wrapper
的引用。- 当
add(3, 5)
执行时:- 先 打印
"开始执行 add,参数:(3, 5) {}"
- 再 运行
add(3, 5)
- 最后 记录
"add 执行完毕,返回值:8"
- 先 打印
输出示例:
开始执行 add,参数:(3, 5) {}
add 执行完毕,返回值:8
8
3. 组合多个装饰器
装饰器可以层层叠加,按顺序执行:
@time_logger
@log_decorator
def multiply(a, b):time.sleep(0.5)return a * bprint(multiply(2, 3))
执行流程:
- 先 执行
log_decorator
- 再 执行
time_logger
- 最后 执行
multiply
输出示例:
开始执行 multiply,参数:(2, 3) {}
函数执行完毕
multiply 执行时间: 0.500123 秒
multiply 执行完毕,返回值:6
6
四、JavaScript 装饰器示例
1. 计时装饰器
function timeLogger(func) {return function (...args) {const startTime = Date.now();const result = func(...args); // 执行原函数const endTime = Date.now();console.log(`${func.name} 执行时间: ${(endTime - startTime) / 1000} 秒`);return result;};
}function slowFunction() {console.log("函数执行中...");for (let i = 0; i < 1e9; i++) {} // 模拟耗时操作console.log("函数执行完毕");
}const wrappedFunction = timeLogger(slowFunction);
wrappedFunction();
执行流程:
timeLogger(slowFunction)
返回wrapper
,wrappedFunction
变成wrapper
。- 执行
wrappedFunction()
时:- 先 记录
startTime
。 - 再 执行
slowFunction()
。 - 最后 计算执行时间并打印。
- 先 记录
输出示例:
函数执行中...
函数执行完毕
slowFunction 执行时间: 2.345 秒
2. ES7+ 修饰器(Decorator)
在现代 JavaScript 中,可以使用 @decorator
语法:
function logDecorator(target, key, descriptor) {const originalMethod = descriptor.value;descriptor.value = function (...args) {console.log(`执行 ${key},参数:`, args);const result = originalMethod.apply(this, args);console.log(`返回值:`, result);return result;};return descriptor;
}class MathOperations {@logDecoratoradd(a, b) {return a + b;}
}const math = new MathOperations();
math.add(3, 5);
输出示例:
执行 add,参数:[3, 5]
返回值:8
五、装饰器模式 vs 代理模式
对比项 | 装饰器模式(Decorator) | 代理模式(Proxy) |
---|---|---|
作用 | 增强对象功能 | 控制对象访问 |
是否修改原对象 | ❌ 不修改 | ❌ 只代理,通常不修改 |
是否拦截请求 | ❌ 不拦截 | ✅ 代理可拦截请求 |
典型应用 | Vue reactive() 、日志、计时 | API 代理、缓存、权限控制 |
六、总结
- 装饰器模式用于动态扩展对象功能,而不修改其原始代码。
- 装饰器本质是一个高阶函数,接收函数/类作为参数,返回一个增强版本。
- Python
@decorator
语法糖让装饰器更直观,JavaScript 也可以使用@decorator
(ES7+)。 - 适用于日志记录、权限控制、缓存优化等场景。
🚀 掌握装饰器模式,让你的代码更优雅、更灵活!
相关文章:
【设计模式】深入解析装饰器模式(Decorator Pattern)
深入解析装饰器模式(Decorator Pattern) 一、装饰器模式的核心概念 装饰器模式是一种结构型设计模式,用于动态地给对象添加新功能,而不改变其原始代码。 1. 为什么需要装饰器? 避免继承带来的类爆炸问题࿱…...
K8S集群新增和删除Node节点(K8s Cluster Adds and Removes Node Nodes)
实战:在已有K8S集群如何新增和删除Node节点 在Kubernetes (K8S) 集群中,Node节点是集群中的工作节点,它们运行着容器的实际实例。管理K8S集群中的Node节点,包括新增和删除节点,是一个常见且重要的操作,可以…...
2503C++,C++标准的执行
最优雅的应该是c26刚刚引入的std::execution,通过sender/receiver模型和常用的异步算法来简化调用异步逻辑,还可随时改成协程. #include <stdexec/execution.hpp> #include <exec/static_thread_pool.hpp> int main() {exec::static_thread_pool pool(3);auto sch…...
nodejs中实现一个自定义的require方法
1.前言 大家对nodejs中的require方法应该不会陌生,这个方法可以用来导入nodejs的内置模块,自定义模块,第三方模块等,使用频率非常高,那么这个方法内部是如何实现的呢?本篇文章就是从头到尾拆分实现流程,最终实现一个自定义的require方法的 2.前置操作 导入所需的nodejs内置…...
vscode/cursor中python运行路径设置 模块导入问题
vscode/cursor中python运行路径设置 ## 文件路径设置 问题描述 pycharm的项目用cursor运行,出现目录找不到 后来利用 os.getcwd(),经过打印调试发现是IDE的本身配置问题 pycharm中,os.getcwd()默认打开当前脚本所在目录 vscode/cursor中…...
Spring学习笔记05——Spring Boot的文件结构2(POJO类)
在Spring Boot项目中,将Entity、DTO、VO放在POJO子模块中是一种常见的分层设计,它们各自承担不同的职责,通过一个通俗的例子来解释它们的作用: POJO(Plain Old Java Object)是指普通的、简单的Java对象&am…...
html和css 实现元素顺时针旋转效果(椭圆形旋转轨迹)
一 实现效果 二 实现代码 我自己是用react写的。 1. react 代码如下: import React from "react"; import styles from "./index.less";export default () > {return <div className{styles.containers}><div className{styles.c…...
C# 的Lambda表达式常见用法和示例
C# 的 Lambda 表达式是一种强大的语法糖,能够极大简化代码并增强灵活性。以下是它的主要功能和应用场景,结合具体示例说明: 1. 简化委托实例化 Lambda 可以快速定义委托(如 Func、Action),无需显式…...
2024年数维杯数学建模C题天然气水合物资源量评价解题全过程论文及程序
2024年数维杯数学建模 C题 天然气水合物资源量评价 原题再现: 天然气水合物(Natural Gas Hydrate/Gas Hydrate)即可燃冰,是天然气与水在高压低温条件下形成的类冰状结晶物质,因其外观像冰,遇火即燃&#…...
Qt中10倍提升动态截屏及渲染60帧/秒
Qt中10倍提升动态截屏及渲染60帧/秒 理解模态窗口和非模态窗口 在C中,窗口的**模态(Modal)和非模态(Modeless)**显示是两种不同的对话框或窗口行为模式,主要区别体现在用户交互和程序流程控制上。以下是它…...
OpenCV 基础全方位剖析:夯实计算机视觉开发根基
在计算机视觉的广袤领域中,OpenCV 是一座极为关键的里程碑。无论是在前沿的学术研究,还是在蓬勃发展的工业界,OpenCV 凭借其强大的功能与高效的性能,为开发者提供了丰富的图像处理和计算机视觉算法,助力无数项目落地。…...
Java试题
试题: 解析 1-5: 5: 6: 7: 8: 9: 10: 11: 12: 13:...
基于 arco 的 React 和 Vue 设计系统
arco 是字节跳动出品的企业级设计系统,支持React 和 Vue。 安装模板工具 npm i -g arco-cli创建项目目录 cd someDir arco init hello-arco-pro? 请选择你希望使用的技术栈React❯ Vue? 请选择一个分类业务组件组件库Lerna Menorepo 项目❯ Arco Pro 项目看到以…...
解密细胞衰老与溶解:AbMole助力胰腺癌研究新突破
近日,一项由德国罗斯托克大学医学中心的研究团队完成的研究,在探索胰腺癌细胞衰老与溶解的复杂机制上取得了重要进展。这项研究不仅深化了我们对胰腺癌生物学特性的理解,更为未来的研究开辟了新的方向。而在这场科学探索中,AbMole…...
罗德与施瓦茨FSU8,频谱分析仪
罗德与施瓦茨FSU8频谱分析仪 R&S FSU系列频谱仪是动态范围、相位噪声、电平精度和分辨率带宽等频谱仪指标,所有这些指标也是用户设计、测量和生产下一代无线通讯元件的重要保障。出色的表现能力 频率范围:从20Hz开始,分别到3.6 GHz, 8…...
【零基础JavaScript入门 | Day7】三大交互案例深度解析|从DOM操作到组件化开发
【零基础JavaScript入门 | Day7】三大交互案例深度解析|从DOM操作到组件化开发 🌟今日知识图谱: ✅ 事件驱动编程 → 按钮交互与定时器控制 ✅ 组件化思维 → 可复用UI模块开发 ✅ 用户体验优化 → 动画与状态反馈设计 ✅ 工程化实践 → 代码…...
[BalticOI 2009] Radio Transmission 无线传输
题目来自洛谷网站: KMP思路: 结论:n - ne[n] 模拟样例,如下图所示: 代码: #include<bits/stdc.h> #define int long long using namespace std; const int N 1e620;int n; char s[N]; int ne[N];…...
SvelteKit 最新中文文档教程(10)—— 部署 Cloudflare Pages 和 Cloudflare Workers
前言 Svelte,一个语法简洁、入门容易,面向未来的前端框架。 从 Svelte 诞生之初,就备受开发者的喜爱,根据统计,从 2019 年到 2024 年,连续 6 年一直是开发者最感兴趣的前端框架 No.1: Svelte …...
32位汇编:MASM32环境搭建与汇编窗口程序
引言 “汇编语言”是计算机底层的编程语言,直接操作硬件资源。32位汇编相比16位汇编在寄存器宽度、内存寻址和指令集等方面有了显著提升。本文将带你从零开始搭建32位汇编开发环境,并编写第一个窗口程序。 1. 环境搭建 1.1 下载MASM32 MASM32是一个专门…...
如何在 `php-fpm` 启动后自动运行自定义命令
如何在 php-fpm 启动后自动运行自定义命令 在使用 php-fpm 时,有时需要在 php-fpm 启动后自动运行一些自定义命令,例如启动一个 Web 应用程序。本文将详细介绍如何通过 systemd 的 ExecStartPost 指令实现这一功能,并记录解决过程中遇到的问…...
26考研——图_图的代码实操(6)
408答疑 文章目录 五、图的代码实操图的存储邻接矩阵结构定义初始化插入顶点获取顶点位置在顶点 v1 和 v2 之间插入边获取第一个邻接顶点获取下一个邻接顶点显示图 邻接表结构定义初始化图插入顶点获取顶点位置在顶点 v1 和 v2 之间插入边获取第一个邻接顶点获取下一个邻接顶点…...
栈资源释放示例
import java.util.EmptyStackException; /** * 栈资源释放示例类 * 栈资源在编程中通常指两种概念:1) 内存中的调用栈空间(存储方法调用和局部变量),由系统自动管理 * 2) 自定义的栈数据结构(如链表实现的栈对象&…...
【Qt 01】cmake搭建Qt VS2019开发环境
文章目录 准备第一步:安装Qt5 库第二步:编写CMakeLists.txt第三步:编写批处理文件其他配置Qt VS Toolsqt 源码调试配置 疑问初始的.ui, .qrc文件如何生成? 准备 编码:vscode debug: vs2019 Qt vs Tools 插件 c包管理…...
余弦退火算法与学习率预热
余弦退火算法与学习率预热 总述: (1)标准余弦退火算法(Cosine Annealing)是通过单次的余弦曲线调整学习率,在一个周期内让学习率从一个最大值平滑下降η_max到最小值η_min,这种调整模式是非周…...
ref vs reactive,watch vs watchEffect的区别与使用场景
一、ref 与 reactive 的核心区别 使用场景 ref:处理基本数据类型;需要重新赋值对象(如从api获取新数据) reactive:处理复杂的嵌套对象或者数组,不需要整体替换,直接访问属性(避免频繁写.value&…...
SQLark SQL编辑器秘籍,编写高效SQL查询
SQLark 是一款功能强大的数据库开发和管理工具,用于快速查询、创建和管理不同类型的数据库系统,支持达梦、Oracle 和 MySQL 数据库。SQLark内置的 SQL 编辑器,基于语法解析,集成智能提示、实时语法检查及语法高亮等功能࿰…...
LLM之RAG实战(五十二)| 如何使用混合搜索优化RAG 检索
在RAG项目中,大模型生成的参考内容(专业术语称为块)来自前一步的检索,检索的内容在很大程度上直接决定了生成的效果,因此检索对于RAG项目至关重要,最常用的检索方法是关键字搜索和语义搜索。本文将分别介绍…...
如何基于ios部署Deep Seek?
在 iOS 上部署深度学习模型(如 DeepSeek 或其他自定义模型)通常需要将模型转换为 iOS 支持的格式(如 Core ML),并通过代码集成到应用中。以下是详细步骤: 1. 准备模型 模型训练 确保你的模型已训练完成&…...
spring security 使用的过滤器还是拦截器
spring security 使用的过滤器还是拦截器 Spring Security 是一个强大的安全框架,用于保护 Java 应用程序。它主要使用过滤器(Filters)来实现安全功能,而不是拦截器(Interceptors)。不过,它也提…...
新四化驱动,如何构建智能汽车的“全场景”可进化互联网络?
在智能化、电动化、网联化、共享化的时代浪潮中,汽车正从单纯的 “机械产品” 进化为先进的 “移动智能终端”。 在软件定义汽车(SDV)的崭新时代,每一次 OTA 升级的顺利完成、每一秒自动驾驶的精准决策、每一帧车载娱乐交互的流畅…...
Rust从入门到精通之进阶篇:17.宏编程基础
宏编程基础 宏是 Rust 中强大的元编程工具,允许你编写可以生成其他代码的代码。与函数不同,宏在编译时展开,可以实现更灵活的代码生成和重用模式。在本章中,我们将探索 Rust 的宏系统,包括声明宏和过程宏的基础知识。 宏与函数的区别 在深入宏编程之前,让我们先了解宏…...
MySQL基础语法
目录 一、数据定义语言(DDL) 1. 创建数据库 2. 删除数据库 3. 创建表 4. 删除表 5. 查看表结构 6. 修改表结构 二、数据操作语言(DML) 1. 插入数据 2. 更新数据 3. 删除数据 三、数据查询语言(DQLÿ…...
数据结构练习二习题
第七题: #include<bits/stdc.h> using namespace std;#define MaxSize 100 // 定义顺序表的最大长度typedef int ElemType; // 定义元素类型为inttypedef struct {ElemType data[MaxSize]; // 存储数据的数组int length; // 当前顺序表的长度…...
Flutter 中 GetX 的优缺点及常见问题解决方案
在 Flutter 生态中,GetX 凭借其轻量级、高效的特性,成为众多开发者青睐的状态管理与路由解决方案。然而,任何工具都有其适用场景与局限性。 一、GetX 的核心优势 1. 极简开发体验 GetX 通过响应式语法糖(如Rx和Obx)…...
Qt开发:QColorDialog的使用
文章目录 一、QColorDialog的基本介绍二、QColorDialog的基本用法三、完整示例 一、QColorDialog的基本介绍 QColorDialog 继承自 QDialog,用于让用户选择颜色。它可以以模态或非模态的方式运行,并支持 QColor 作为输入和输出。 二、QColorDialog的基本…...
【计算机网络编码与调制】
文章目录 一、前言二、编码与调制的基本概念1. 编码常见编码类型: 2. 调制常见调制类型: 三、常见编码技术1. NRZ编码(Non-Return to Zero)特点: 2. 曼彻斯特编码(Manchester)特点: …...
<数据集>手势识别数据集<目标检测>
数据集下载链接https://download.csdn.net/download/qq_53332949/90529961数据集格式:VOCYOLO格式 图片数量:10829张 标注数量(xml文件个数):10829 标注数量(txt文件个数):10829 标注类别数:26 标注类别名称&…...
魔法测试:用本地多模态大模型(Qwen2.5-VL)将PDF转为Markdown文档
安装咒语清单 📜 pip install -U vllm qwen-vl-utils pdf2image pip install githttps://github.com/huggingface/transformers accelerate(温馨提示:念咒前请检查你的魔法棒(Python环境)是否兼容~) 第一阶段:PDF变…...
【渗透测试】Fastjson 反序列化漏洞原理(二)
目录 反序列化漏洞结合 RMI 攻击详解一、RMI 和 JNDI 的基础知识(1) RMI 简介(2) JNDI 简介 二、漏洞利用的核心原理1. 构造恶意 JSON 数据2. 设置恶意 RMI 服务3. 加载并执行恶意代码 三、具体利用过程环境准备攻击步骤1. 构造恶意 JSON 数据2. 设置恶意 RMI 服务3. 部署恶意类…...
[c语言日寄MAX]深度解析:大小端字节序
【作者主页】siy2333 【专栏介绍】⌈c语言日寄MAX⌋:这是一个专注于C语言刷题的专栏,精选题目,搭配详细题解、拓展算法。从基础语法到复杂算法,题目涉及的知识点全面覆盖,助力你系统提升。无论你是初学者,还…...
凝视型高光谱相机:钻石光谱分析研究与应用
钻石作为一种珍贵的宝石,其颜色和光谱特征一直是宝石学研究的重要方向。中达瑞和测试部样品科对六种不同颜色的钻石样品进行了高光谱成像测试,旨在分析它们在光谱下的差异和可分性。本文将详细介绍此次测试的过程、结果及其意义。 测试样品与设备 此次…...
2.基于多线程的TCP服务器实现
在我们预想中,服务器端应该能够同时与多个客户端建立连接并进行网络通信。然而,在之前的代码中,服务器实现只支持单一连接,因为在处理连接时,主线程会被accept()、read()或write()等方法阻塞,导致无法响应新…...
自动驾驶01 激光雷达原理
部分转载于 原文链接:https://blog.csdn.net/qq_45193988/article/details/143982588 1. 什么是激光雷达 激光雷达LiDAR的全称为Light Detection and Ranging 激光探测和测距。 激光雷达的工作原理:对红外光束Light Pluses发射、反射和接收来探测物体。…...
基础场景-------------------(5)重载和重写的区别
重写(Override) 从字面上看,重写就是 重新写一遍的意思。其实就是在子类中把父类本身有的方法重新写一遍。子类继 承了父类原有的方法,但有时子类并不想原封不动的继承父类中的某个方法,所以在方法名,参数列 表,返回…...
基于ssm的微博网站(全套)
进入信息时代以来,很多数据都需要配套软件协助处理,这样可以解决传统方式带来的管理困扰。比如耗时长,成本高,维护数据困难,数据易丢失等缺点。本次使用数据库工具MySQL和编程技术JSP开发的微博网站,可以实…...
Java并发编程从入门到实战:同步、异步、多线程核心原理全解析
《Java并发编程从入门到实战:同步、异步、多线程核心原理全解析》 一、多线程基础认知(从单核到多核的进化) 1.1 什么是线程? 线程是程序执行的最小单元,一个进程可以包含多个线程。例如浏览器同时下载文件࿰…...
《引流获客》总结
第一章 入门篇 理解爆款打法和引流打法的区别 爆款打法:刷播放量,制作“爆款”视频/文案。 引流打法:刷有效转化,不在意播放量,而注重有多少观众被引流成为客户,完成消费,即成交转化。 定理&…...
基于Python的机器学习入门指南
在当今数字化时代,机器学习(Machine Learning)已经成为科技领域中最热门的话题之一。它不仅改变了我们对数据的理解和处理方式,还在许多行业中得到了广泛应用,如金融、医疗、交通等。Python作为一门强大的编程语言&…...
【蓝桥杯每日一题】3.25
🏝️专栏: 【蓝桥杯备篇】 🌅主页: f狐o狸x “OJ超时不是终点,是算法在提醒你该优化时间复杂度了!” 目录 3.25 差分数组 一、一维差分 题目链接: 题目描述: 解题思路:…...
Kubernetes高级应用之-重启策略
一、介绍+扩展应用(涉及的高级资源在后续会写出来) # Kubernetes Pod重启策略(RestartPolicy)全面解析 ## 一、重启策略的核心价值与重要性 在Kubernetes集群中,Pod重启策略(RestartPolicy&a…...