当前位置: 首页 > news >正文

上位机开发中的设计模式(3):装饰器模式

1.装饰器模式(Decorator Pattern)的定义

装饰器模式是一种结构型设计模式,其核心思想是:

  1. 动态地给对象添加额外功能,而不改变其原有结构
  2. 通过**包装(wrapping)**原始对象来提供增强功能
  3. 遵循开闭原则(对扩展开放,对修改关闭)

UML核心组件

  • Component:定义原始对象的接口
  • ConcreteComponent:原始对象的具体实现
  • Decorator:持有Component引用并实现相同接口
  • ConcreteDecorator:具体的装饰器实现

2.背景

我在搭建python测试框架时,无论是unittest还是pytest,均提供了装饰器模式,对function加入装饰器声明包装,使得function成为待使用的case

3.Python装饰器(Decorator)的实现

Python中的装饰器语法(@decorator)是装饰器模式的一种语法糖实现,但二者并不完全等同:

def decorator(func):def wrapper(*args, **kwargs):print("Before function")  # 添加新功能result = func(*args, **kwargs)  # 调用原函数print("After function")   # 添加新功能return resultreturn wrapper@decorator
def original_function():print("Original function")

4.测试框架中的"装饰器模式"应用

1) pytest的实现方式
@pytest.mark.slow
def test_function():pass
  • 本质pytest.mark.slow是一个函数装饰器,它给测试函数添加了元数据
  • 装饰器模式体现
    • 原始组件:测试函数
    • 装饰器:pytest.mark系统
    • 新增功能:添加标记(mark)信息到测试函数
2) unittest的实现方式
@unittest.skip("reason")
def test_method(self):pass
  • 本质unittest.skip是一个类装饰器(实际是描述符协议实现)
  • 装饰器模式体现
    • 原始组件:测试方法
    • 装饰器:unittest.skip等装饰器
    • 新增功能:改变测试方法的执行行为(如跳过测试)

5.为什么说这是装饰器模式?

虽然测试框架中的装饰器使用看起来像是简单的语法装饰器,但它们实际上符合装饰器模式的核心思想:

  1. 不修改原测试函数/方法:保持原始测试逻辑不变
  2. 动态添加功能
    • pytest:添加标记、参数化、fixture依赖等
    • unittest:添加跳过、预期失败等行为
  3. 多层包装能力
    @pytest.mark.slow
    @pytest.mark.parametrize("input", [1,2,3])
    def test_func(input): pass
    

6.上位机中的装饰器模式

下面展示一个在QT上位机应用中使用装饰器模式的完整示例。这个例子模拟了一个数据可视化系统,可以动态地给数据处理器添加不同的功能(如日志记录、数据验证、加密等)。

场景描述

数据处理器可以动态添加以下功能:

  1. 日志记录功能
  2. 数据验证功能
  3. 数据加密功能

实现代码

#include <QCoreApplication>
#include <QDebug>
#include <QString>
#include <memory>// 抽象组件接口
class DataProcessor {
public:virtual ~DataProcessor() = default;virtual QString process(const QString& data) = 0;
};// 具体组件 - 核心数据处理功能
class CoreDataProcessor : public DataProcessor {
public:QString process(const QString& data) override {qDebug() << "Core processing data:" << data;// 模拟核心处理逻辑return data.toUpper();}
};// 抽象装饰器
class DataProcessorDecorator : public DataProcessor {
protected:std::unique_ptr<DataProcessor> processor;
public:DataProcessorDecorator(std::unique_ptr<DataProcessor> processor): processor(std::move(processor)) {}
};// 具体装饰器 - 日志记录
class LoggingDecorator : public DataProcessorDecorator {
public:using DataProcessorDecorator::DataProcessorDecorator;QString process(const QString& data) override {qDebug() << "Logging: Before processing -" << data;QString result = processor->process(data);qDebug() << "Logging: After processing -" << result;return result;}
};// 具体装饰器 - 数据验证
class ValidationDecorator : public DataProcessorDecorator {
public:using DataProcessorDecorator::DataProcessorDecorator;QString process(const QString& data) override {if(data.isEmpty()) {qWarning() << "Validation failed: Empty data";return "";}return processor->process(data);}
};// 具体装饰器 - 数据加密
class EncryptionDecorator : public DataProcessorDecorator {QString encrypt(const QString& data) {// 简单加密示例 - 实际应用中替换为真正的加密算法QString result;for(QChar ch : data) {result.append(QChar(ch.unicode() + 1));}return result;}QString decrypt(const QString& data) {// 简单解密QString result;for(QChar ch : data) {result.append(QChar(ch.unicode() - 1));}return result;}public:using DataProcessorDecorator::DataProcessorDecorator;QString process(const QString& data) override {QString encrypted = encrypt(data);qDebug() << "Encrypted data:" << encrypted;QString processed = processor->process(encrypted);return decrypt(processed);}
};// QT上位机应用示例
int main(int argc, char *argv[]) {QCoreApplication a(argc, argv);// 创建基础处理器std::unique_ptr<DataProcessor> processor = std::make_unique<CoreDataProcessor>();// 动态添加功能bool enableLogging = true;bool enableValidation = true;bool enableEncryption = true;if(enableLogging) {processor = std::make_unique<LoggingDecorator>(std::move(processor));}if(enableValidation) {processor = std::make_unique<ValidationDecorator>(std::move(processor));}if(enableEncryption) {processor = std::make_unique<EncryptionDecorator>(std::move(processor));}// 模拟上位机数据处理QString testData = "Hello QT Decorator Pattern";qDebug() << "Original data:" << testData;QString result = processor->process(testData);qDebug() << "Final result:" << result;// 测试空数据验证qDebug() << "\nTesting empty data validation:";processor->process("");return a.exec();
}
输出示例
Original data: "Hello QT Decorator Pattern"
Encrypted data: "Ifmmp!RU!Efdpsojps!Qbssfsu"
Logging: Before processing - "Ifmmp!RU!Efdpsojps!Qbssfsu"
Core processing data: "Ifmmp!RU!Efdpsojps!Qbssfsu"
Logging: After processing - "IFMMP!RU!EFDPSOJPS!QBSSFSU"
Final result: "HELLO QT DECORATOR PATTERN"Testing empty data validation:
Validation failed: Empty data
关键点解析
  1. 组件接口DataProcessor 定义了核心接口
  2. 具体组件CoreDataProcessor 实现基础功能
  3. 装饰器基类DataProcessorDecorator 持有被装饰对象的指针
  4. 具体装饰器
    • LoggingDecorator 添加日志功能
    • ValidationDecorator 添加数据验证
    • EncryptionDecorator 添加加密/解密功能
  5. 动态组合:在运行时根据需要组合各种功能
QT实际应用场景
  1. 通信协议栈:可以动态添加CRC校验、数据压缩、加密等层
  2. UI组件增强:为基本控件添加动画、阴影等效果
  3. 数据处理流水线:动态组合不同的数据处理算法
  4. 插件系统:通过装饰器模式动态扩展功能

这种模式在QT上位机开发中特别有用,因为它允许灵活地组合功能而不需要修改现有代码,符合开闭原则。

http://www.xdnf.cn/news/1039015.html

相关文章:

  • SCADE Suite / Scade 6 官方参考材料介绍
  • 理论物理学中的规范场论与全息原理
  • 6.14星期六休息一天
  • 单片机 - STM32F407 ADC 模式详解:单次转换、连续转换、扫描模式、非扫描模式
  • 项目 【Http】第二讲---CGI机制的介绍
  • 【Unity】使用 C# SerialPort 进行串口通信
  • Java 常用类 Math:从“如何生成随机密码”讲起
  • LCEL:LangChain 表达式语言详解与测试工程师的实践指南
  • 钉钉机器人-自定义卡片推送快速入门
  • wget批量调用shell脚本
  • C#里与嵌入式系统W5500网络通讯(7)
  • 有关Spring事务的传播机制
  • 23. [实用] 扣子(coze)教程 | 小程序UI设计进阶(五)只此一家,标签组件攻略
  • 互联网校招腾讯26届校招暑期实习综合素质测评答题攻略及真题题库
  • monorepo使用指北
  • 123数字人视频剪辑源码搭建部署/数字人视频创作技术开发
  • React配置别名路径完整指南
  • uniapp的app项目,在华为pad上运行,页面显示异常
  • 动目标显示处理解析六(重频参差扩展盲速)
  • static的三种作用
  • 【C++】模拟实现map和set
  • [Linux入门] Linux磁盘管理与文件系统
  • YOLOv3 中的 IoU 计算详解
  • 在Ubuntu linux终端写文件的方法
  • FFmpeg开发笔记(七十一)使用国产的QPlayer2实现双播放器观看视频
  • 【Zephyr 系列 25】多芯片协同设计:主控 + BLE + LoRa 芯片的统一调度与消息系统
  • 什么是泛型,如何使用它?
  • 动态组件(component)的高级使用
  • PL端DDR3读写(1)
  • 转换专家从格式转换到创意剪辑的全链路解决方案