穿了马甲你就牛逼了:装饰者模式解析

2024-03-19 06:18
文章标签 模式 解析 装饰 马甲

本文主要是介绍穿了马甲你就牛逼了:装饰者模式解析,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本文是设计模式系列的第三篇文章,今天主要学习装饰者模式。

不知道大家没有有这样一种感觉,在看书学习时,感觉都看懂了,可是过一段时间就忘,因此我们从开头就先问自己几个问题,过一段时间就回过头了复习下这几个问题,从而巩固学到的知识,在你的大脑中将这些知识点串起来。 希望能不断反复的思考,将点成线,最终形成知识块,消化掉它。

带着问题学习

  1. 什么是装饰者模式?
  2. 什么场景下需要使用装饰者模式?
  3. 如何实现装饰者模式?
  4. 常用框架或源码中有哪些案例可以体现?

装饰者模式的概念

我们先来看看装饰者模式的说明:

「装饰者模式」 动态的将责任附加到对象上。若要扩展功能,装饰者提供了比继承更有弹性的替代方案。

「新的设计原则」:类应该对扩展开放对修改关闭。

上面的定义虽然说明了装饰者模式的 “角色”,但是没有说明怎么在我们的实现中实际 “应用”它。

接下来我们学习下它的类图,结合着类图在仔细分析它:

从上面的类图,我们再来理解下装饰者模式

  • 装饰者和被装饰对象有相同的超类型
  • 你可以用一个或多个装饰者包装一个对象(component的具体组件)
  • 既然装饰者和被装饰对象有相同的超类型,所以在任何需要原始对象(被包装)的场合,可以用装饰过的对象代替它
  • 「装饰者可以在所委托被装饰者的行为 之前或者之后,加上自己的行为,已达到特定的目的」,后面我们在如何使用中会明确的看到这点
  • 对象可以在任何时候被装饰,所以可以在运行时动态地、不限量地用你喜欢的装饰者来装饰对象

使用场景

现在有一个咖啡店,可以售卖几种不同的咖啡,比如 摩卡、卡布奇诺、玛奇朵、康巴纳等。而且每一种具体的咖啡还可以添加不同的调料,比如奶泡、焦糖、豆浆、摩卡等,要求加不同调料最终价格不同,这个场景如果使用 OO 思想来设计,你会怎么做呢?

毫无疑问这个场景非常适合装饰者模式,首先被装饰者就是我们的咖啡品种,而装饰者就是我们的不同调料,这样在计算价格时可以一层层去委托最终得到结果,话不多说来看如果用装饰者模式来装饰我们的咖啡。

看了上面我们举例的星巴克咖啡的例子,有没有和装饰者模式的类图框架对应呢? 大家可以仔细思考下。

巩固扩展

利用上面你的装饰者模式现在出一个场景,我们测试下到底能不能在不改动现有代码的前提下实现,体验下设计模式的魅力。

「点一杯双倍摩卡豆浆奶泡拿铁咖啡?」

是不是很方便,我们利用装饰者模式进行组合扩展,体验到魅力了吧。。

具体实现

下面一我们一起结合上面的例子,看下代码实现:

// 饮料的基础类,即component
public abstract class Beverage {String description = "Unknown Beverage";public String getDescription() {return description;  }  // cost必须在子类实现  public abstract double cost(); } 

调料抽象类,即装饰者类:

// 调料抽象类即装饰者类,这个类必须要能替换 Beverage,所以要继承自 Beverage 类
public abstract class CondimentDecorator extends Beverage {public abstract String getDescription();
}

现在有了基类,下面是一个具体的饮料:

// 蓝山
public class BlueMountainCoffee extends Beverage {public BlueMountainCoffee() {description = "BlueMountainCoffee";}@Override  public double cost() {  return 0;  } }  // 卡布奇诺 public class Cappuccino extends Beverage {  public Cappuccino() {  description = "Cappuccino";  }  @Override  public double cost() {  return 23;  } }  // 意式浓缩咖啡 public class Espresso extends Beverage {  public Espresso() {  description = "Espresso";  }  @Override  public double cost() {  return 25;  } } // 拿铁 public class Latte extends Beverage {  public Latte() {  description = "Latte";  }  @Override  public double cost() {  return .89;  } } 

现在已经有了具体组件和抽象组件,对比装饰者模式类图我们实现具体的装饰者:

// 摩卡是一个装饰者,所以扩展自 CondimentDecorator
public class Mocha extends CondimentDecorator {Beverage beverage;public Mocha(Beverage beverage) {this.beverage = beverage;}  @Override  public String getDescription() {  return beverage.getDescription() + ",Mocha";  }   // 首先调用委托被装饰者对象,以计算价钱,然后再加上Mocha价钱  @Override  public double cost() {  return .20 + beverage.cost();  } } // 豆浆 public class Soy extends CondimentDecorator {  Beverage beverage;  public Soy(Beverage beverage) {  this.beverage = beverage;  }  @Override  public String getDescription() {  return beverage.getDescription() + ",Soy";  }  @Override  public double cost() {  return 2.0 + beverage.cost();  } } 

接下来就是展示装饰者模式魅力的时候:

// 测试类
public class StarbuzzCoffee {public static void main(String[] args) {// 一杯Espresso,不加调料Beverage beverage = new Espresso();System.out.println(beverage.getDescription() + "$" + beverage.cost());  // 一杯加摩卡和豆浆的蓝山咖啡  Beverage beverage1 = new BlueMountainCoffee();  beverage1 = new Mocha(beverage1);  beverage1 = new Soy(beverage1);  System.out.println(beverage1.getDescription() + "$" + beverage1.cost());  } } 

目前我们创建对象还都是硬编码 new 出来的,不太友好,随着后续我们学习了工厂模式就好了,持续学习吧。

现实中的装饰者

这块列举些平时用到的 jdk 中的装饰者模式体现

Java I/O

列出的顺序是从装饰者 -> 被装饰者

LineNumberInputStream -> BufferedInputStream -> FileInputStream

一目了然吧,这个和我们上面讲的装饰者模式类图基本上是一致的,相信在你再次阅读 Jvaa I/O 包中的类时,你一定会发出 “哇” 的一声惊叹。


全文完!fighting

原创真心不易,希望你能帮我个小忙呗,如果本文内容你觉得有所收获,请帮忙点个“在看”呗,或者转发分享让更多的小伙伴看到。

这篇关于穿了马甲你就牛逼了:装饰者模式解析的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/825062

相关文章

深度解析Spring Security 中的 SecurityFilterChain核心功能

《深度解析SpringSecurity中的SecurityFilterChain核心功能》SecurityFilterChain通过组件化配置、类型安全路径匹配、多链协同三大特性,重构了Spri... 目录Spring Security 中的SecurityFilterChain深度解析一、Security

全面解析Golang 中的 Gorilla CORS 中间件正确用法

《全面解析Golang中的GorillaCORS中间件正确用法》Golang中使用gorilla/mux路由器配合rs/cors中间件库可以优雅地解决这个问题,然而,很多人刚开始使用时会遇到配... 目录如何让 golang 中的 Gorilla CORS 中间件正确工作一、基础依赖二、错误用法(很多人一开

Mysql中设计数据表的过程解析

《Mysql中设计数据表的过程解析》数据库约束通过NOTNULL、UNIQUE、DEFAULT、主键和外键等规则保障数据完整性,自动校验数据,减少人工错误,提升数据一致性和业务逻辑严谨性,本文介绍My... 目录1.引言2.NOT NULL——制定某列不可以存储NULL值2.UNIQUE——保证某一列的每一

深度解析Nginx日志分析与499状态码问题解决

《深度解析Nginx日志分析与499状态码问题解决》在Web服务器运维和性能优化过程中,Nginx日志是排查问题的重要依据,本文将围绕Nginx日志分析、499状态码的成因、排查方法及解决方案展开讨论... 目录前言1. Nginx日志基础1.1 Nginx日志存放位置1.2 Nginx日志格式2. 499

MySQL CTE (Common Table Expressions)示例全解析

《MySQLCTE(CommonTableExpressions)示例全解析》MySQL8.0引入CTE,支持递归查询,可创建临时命名结果集,提升复杂查询的可读性与维护性,适用于层次结构数据处... 目录基本语法CTE 主要特点非递归 CTE简单 CTE 示例多 CTE 示例递归 CTE基本递归 CTE 结

Spring Boot 3.x 中 WebClient 示例详解析

《SpringBoot3.x中WebClient示例详解析》SpringBoot3.x中WebClient是响应式HTTP客户端,替代RestTemplate,支持异步非阻塞请求,涵盖GET... 目录Spring Boot 3.x 中 WebClient 全面详解及示例1. WebClient 简介2.

在MySQL中实现冷热数据分离的方法及使用场景底层原理解析

《在MySQL中实现冷热数据分离的方法及使用场景底层原理解析》MySQL冷热数据分离通过分表/分区策略、数据归档和索引优化,将频繁访问的热数据与冷数据分开存储,提升查询效率并降低存储成本,适用于高并发... 目录实现冷热数据分离1. 分表策略2. 使用分区表3. 数据归档与迁移在mysql中实现冷热数据分

C#解析JSON数据全攻略指南

《C#解析JSON数据全攻略指南》这篇文章主要为大家详细介绍了使用C#解析JSON数据全攻略指南,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录一、为什么jsON是C#开发必修课?二、四步搞定网络JSON数据1. 获取数据 - HttpClient最佳实践2. 动态解析 - 快速

Spring Boot3.0新特性全面解析与应用实战

《SpringBoot3.0新特性全面解析与应用实战》SpringBoot3.0作为Spring生态系统的一个重要里程碑,带来了众多令人兴奋的新特性和改进,本文将深入解析SpringBoot3.0的... 目录核心变化概览Java版本要求提升迁移至Jakarta EE重要新特性详解1. Native Ima

spring中的@MapperScan注解属性解析

《spring中的@MapperScan注解属性解析》@MapperScan是Spring集成MyBatis时自动扫描Mapper接口的注解,简化配置并支持多数据源,通过属性控制扫描路径和过滤条件,利... 目录一、核心功能与作用二、注解属性解析三、底层实现原理四、使用场景与最佳实践五、注意事项与常见问题六