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

装饰模式解析:基本概念和实例教程

目录

  • 装饰模式
    • 装饰模式结构
    • 装饰模式应用场景
    • 装饰模式优缺点
    • 练手题目
      • 题目描述
      • 输入描述
      • 输出描述
      • 题解

装饰模式

装饰模式,又称装饰者模式、装饰器模式,是一种结构型设计模式,允许你通过将对象放入包含行为的特殊封装对象中来为原对象绑定新的行为。

装饰模式结构

在这里插入图片描述

  1. 部件 (Component) 声明封装器和被封装对象的公用接口。
  2. 具体部件 (Concrete Component) 类是被封装对象所属的类。 它定义了基础行为, 但装饰类可以改变这些行为。
  3. 基础装饰 (Base Decorator) 类拥有一个指向被封装对象的引用成员变量。 该变量的类型应当被声明为通用部件接口, 这样它就可以引用具体的部件和装饰。 装饰基类会将所有操作委派给被封装的对象。
  4. 具体装饰类 (Concrete Decorators) 定义了可动态添加到部件的额外行为。 具体装饰类会重写装饰基类的方法, 并在调用父类方法之前或之后进行额外的行为。
  5. 客户端 (Client) 可以使用多层装饰来封装部件, 只要它能使用通用接口与所有对象互动即可。

通用代码结构示例


//部件(设计之禅中提到成绩单)
interface Component{void execute();...
}//具体部件 (4年级成绩单)
class ConcreteComponent implements Component{@Overridepublic void execute(){...}
}//基础装饰类
class BaseDecorator implements Component{private Component c;public BaseDecorator(Component c){this.c=c;}@Overridepublic void execute(){...}
}//具体装饰类
class ConcreteDecorators extends{public ConcreteDecorators(Component c){super(c);}public void extra(){...}@Overridepublic void execute(){this.extra();super.execute();...}
}//客户端
public class Client{Component c = new ConcreteComponent();c = new ConcreteDecorators();c.excute();...
}

装饰模式应用场景

  • 如果你希望在无需修改代码的情况下即可使用对象,且希望在运行时为对象新增额外的行为,可以使用装饰模式。

    装饰能将业务逻辑组织为层次结构, 你可为各层创建一个装饰, 在运行时将各种不同逻辑组合成对象。 由于这些对象都遵循通用接口, 客户端代码能以相同的方式使用这些对象。

  • 如果用继承来扩展对象行为的方案难以实现或者根本不可行, 你可以使用该模式。

    许多编程语言使用 final最终关键字来限制对某个类的进一步扩展。 复用最终类已有行为的唯一方法是使用装饰模式: 用封装器对其进行封装。

在这里插入图片描述

识别方法:装饰可通过以当前类或对象为参数的创建方法或构造函数来识别。

装饰模式优缺点

装饰模式优点:

  • 你无需创建新子类即可扩展对象的行为。
  • 你可以在运行时添加或删除对象的功能。
  • 你可以用多个装饰封装对象来组合几种行为。
  • 单一职责原则。 你可以将实现了许多不同行为的一个大类拆分为多个较小的类。

装饰模式缺点:

  • 在封装器栈中删除特定封装器比较困难。
  • 实现行为不受装饰栈顺序影响的装饰比较困难。
  • 各层的初始化配置代码看上去可能会很糟糕。

练手题目

题目描述

小明喜欢品尝不同口味的咖啡,他发现每种咖啡都可以加入不同的调料,比如牛奶、糖和巧克力。他决定使用装饰者模式制作自己喜欢的咖啡。

请设计一个简单的咖啡制作系统,使用装饰者模式为咖啡添加不同的调料。系统支持两种咖啡类型:黑咖啡(Black Coffee)和拿铁(Latte)。

输入描述

多行输入,每行包含两个数字。第一个数字表示咖啡的选择(1 表示黑咖啡,2 表示拿铁),第二个数字表示要添加的调料类型(1 表示牛奶,2 表示糖)。

输出描述

根据每行输入,输出制作咖啡的过程,包括咖啡类型和添加的调料。

在这里插入图片描述

题解

简单的装饰模式实现。

import java.util.Scanner;// 定义咖啡接口
interface Coffee {void execute();
}// 黑咖啡类,实现咖啡接口
class BrewingBlackCoffee implements Coffee {@Overridepublic void execute() {System.out.println("Brewing Black Coffee");}
}// 拿铁类,实现咖啡接口
class BrewingLatte implements Coffee {@Overridepublic void execute() {System.out.println("Brewing Latte");}
}// 咖啡装饰器抽象类,实现咖啡接口
abstract class Decorator implements Coffee {private Coffee coffee;public Decorator(Coffee coffee) {this.coffee = coffee;}@Overridepublic void execute() {coffee.execute();}
}// 牛奶装饰器类,继承自装饰器类
class MilkDecorator extends Decorator {public MilkDecorator(Coffee coffee) {super(coffee);}@Overridepublic void execute() {super.execute();System.out.println("Adding Milk");}
}// 糖装饰器类,继承自装饰器类
class SugarDecorator extends Decorator {public SugarDecorator(Coffee coffee) {super(coffee);}@Overridepublic void execute() {super.execute();System.out.println("Adding Sugar");}
}public class Main {public static void main(String[] args) {Scanner scanner = new Scanner(System.in);try {String input;while (scanner.hasNextLine()) {input = scanner.nextLine();if (input.equalsIgnoreCase("exit")) {break;}processInput(input);}} catch (NumberFormatException e) {System.out.println("输入格式无效:" + e.getMessage());} finally {scanner.close();}}// 处理输入的方法private static void processInput(String input) {String[] parts = input.split(" ");if (parts.length != 2) {System.out.println("输入格式无效。请提供两个数字,中间用空格分隔。");return;}try {int type1 = Integer.parseInt(parts[0]);int type2 = Integer.parseInt(parts[1]);Coffee coffee = createCoffee(type1);if (coffee == null) {System.out.println("咖啡类型无效。请输入1(黑咖啡)或2(拿铁)。");return;}coffee = decorateCoffee(coffee, type2);if (coffee == null) {System.out.println("装饰类型无效。请输入1(牛奶)或2(糖)。");return;}coffee.execute();} catch (NumberFormatException e) {System.out.println("输入格式无效:两个输入都必须是数字。");}}// 创建咖啡对象的方法private static Coffee createCoffee(int type) {switch (type) {case 1:return new BrewingBlackCoffee();case 2:return new BrewingLatte();default:return null;}}// 添加装饰器的方法private static Coffee decorateCoffee(Coffee coffee, int type) {switch (type) {case 1:return new MilkDecorator(coffee);case 2:return new SugarDecorator(coffee);default:return null;}}
}
http://www.lryc.cn/news/391531.html

相关文章:

  • 211.xv6——3(page tables)
  • yum使用报错:ImportError: /lib64/libxml2.so.2: file too short
  • 【Android面试八股文】你是怎么保证Android设备的时间与服务器时间同步的?(使用NTP和TrueTime方案)
  • 解决Python爬虫开发中的数据输出问题:确保正确生成CSV文件
  • SCI一区TOP|徒步优化算法(HOA)原理及实现【免费获取Matlab代码】
  • Android的activity广播无法接收,提示process gone or crashing原因有可能是那些?
  • 如何将等保2.0的要求融入日常安全运维实践中?
  • 51单片机嵌入式开发:STC89C52环境配置到点亮LED
  • 源代码加密:保护你的数字宝藏
  • Jackson库使用教程
  • 汉王、绘王签字版调用封装
  • 如何在TikTok上获得更多观看量:12个流量秘诀
  • vue模板语法v-html
  • 13 Redis-- 数据一致性模型、MySQL 和 Redis 的数据一致性
  • 启动Nuxt-hub-starter: Failed to initialize wrangler bindings proxy write EOF
  • 技术驱动旅游创新!深度解析景区导览小程序的地图渲染与AR导航技术
  • 二叉树之遍历
  • 【经验贴】如何做好自己的职业规划(技术转项目经理)
  • 【笔记】字符串相似度代码分享
  • AI墓地:738个倒闭AI项目的启示
  • 工程文件参考——CubeMX+LL库+SPI主机 阻塞式通用库
  • LLM - 模型历史
  • Go语言中的时间与日期处理:time包详解
  • Java实现单点登录(SSO)详解:从理论到实践
  • 【leetcode82-91动态规划,91-95多维动态规划】
  • Django学习第四天
  • redis-benchmark 使用
  • 什么是 qobject_cast?
  • Python酷库之旅-第三方库Pandas(001)
  • Firefox 编译指南2024 Windows10篇- 编译Firefox(三)