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

【设计模式-3.5】结构型——装饰器模式

说明:本文介绍结构型设计模式之一的装饰器模式

定义

装饰器模式(Decorator Pattern)也叫作包装器模式(Wrapper Pattern),指再不改变原有对象的基础上,动态地给一个对象添加一些额外的职责。就增加功能来说,装饰器模式相比生成子类更为灵活,属于结构型设计模式。(引自《设计模式就该这样学》P201)

化妆例子

以女孩化妆为例,如下:

(可展示接口,Showable)

/*** 能展示的*/
public interface Showable {/*** 展示方法*/void show();
}

(女生对象,Girl)

/*** 女生对象*/
public class Girl implements Showable {@Overridepublic void show() {System.out.print("女孩素颜的面容");}
}

(装饰类,Decorator)

/*** 装饰类*/
public class Decorator implements Showable {private Showable showable;/*** 构造器注入*/public Decorator(Showable showable) {this.showable = showable;}@Overridepublic void show() {System.out.print("抹上淡淡的粉底【");showable.show();System.out.print("】");}
}

(客户端,Client)

public class Client {public static void main(String[] args) {new Decorator(new Girl()).show();}
}

装饰器类扩展了原对象的功能,像是一种包装,包装后更丰富了

在这里插入图片描述

更进一步

在上述化妆的例子上,更进一步,抽象出一个装饰器类,如下:

(装饰抽象类,Decorator)

/*** 装饰类*/
public abstract class Decorator implements Showable {private Showable showable;/*** 构造器注入*/public Decorator(Showable showable) {this.showable = showable;}@Overridepublic void show() {showable.show();}
}

(粉底类,继承装饰类,FoundationMakeup)

/*** 粉底类*/
public class FoundationMakeup extends Decorator {/*** 构造器注入*/public FoundationMakeup(Showable showable) {super(showable);}@Overridepublic void show() {System.out.print("抹上淡淡的粉底【");super.show();System.out.print("】");}
}

(口红类,继承装饰类,Lipstick)

/*** 口红类*/
public class Lipstick extends Decorator {/*** 构造器注入*/public Lipstick(Showable showable) {super(showable);}@Overridepublic void show() {System.out.print("抹上一层口红【");super.show();System.out.print("】");}
}

(客户端使用,Client,可以自由组合装饰方式)

public class Client {public static void main(String[] args) {// 搭配一new Lipstick(new FoundationMakeup(new Girl())).show();System.out.println();// 搭配二new FoundationMakeup(new Lipstick(new Girl())).show();}
}

运行如下:

在这里插入图片描述

JDK源码体现

装饰器模式,在JDK的IO流设计上有体现,如下:

import java.io.BufferedInputStream;
import java.io.File;
import java.io.FileInputStream;
import java.io.FileNotFoundException;
import java.util.zip.ZipInputStream;public class Client {public static void main(String[] args) throws FileNotFoundException {// 装饰器模式在Java源码的体现File file = new File("./src/main/java/structural/decorator/Client.java");ZipInputStream zipInputStream = new ZipInputStream(new BufferedInputStream(new FileInputStream(file)));}
}

扒一下源码,可见ZipInputStreamBufferedInputStreamFileInputStream都继承自抽象类InputStream,类比一下前面的化妆例子,InputStream也就是装饰抽象类。

  • ZipInputStream -> InflaterInputStream -> FilterInputStream -> InputStream

  • BufferedInputStream -> FilterInputStream -> InputStream

  • FileInputStream -> InputStream

与代理模式的区别

关于装饰器模式与代理模式的区别,这里介绍两本书中给的区别:

代理模式和装饰模式在实现时有些类似,但是代理模式主要是给真是主题类增加一些全新的职责,例如权限控制、缓冲处理、智能引用、远程访问等,这些职责与原有职责不属于同一个问题域。而装饰模式是通过装饰类为具体构件类增加一些相关的职责,是对原有职责的扩展,这些职责属于同一问题域。代理模式和装饰模式的目的也不相同,前者是控制对对象的访问,而后者是为对象动态地增加功能。
(引自《设计模式的艺术》刘伟著,第一版P201)


在《设计模式就该这样学》(P214)中,举了一个不错的例子:

简单来讲,假设现在想租房,那么势必会有一些事务发生:房源搜索、联系房东谈价格等。

假设按照代理模式进行思考,那么小明只需要找到一个房产中介,让他去做房源搜索、联系房东谈价格这些事情,小明只需要等待通知然后付中介费就行了。

而如果采用装饰器模式进行思考,因为装饰器模式强调的是自身功能扩展,也就是说,如果要找房子,小明自身就要增加房源搜索能力扩展、联系房东谈价格能力扩展,通过相应的装饰器,提升自身能力,一个人做完所有的事情。

我的理解,装饰器模式与代理模式都能解决问题,没有非装饰器模式能解决的问题,代理模式也是,区别在于解决问题的角度:从内到外,是装饰器模式,从外到内,是代理模式。

总结

本文介绍了结构型设计模式中的装饰器模式,参考《设计模式就该这样学》、《秒懂设计模式》、《设计模式的艺术》三本书,其中化妆的例子来自《秒懂设计模式》。

http://www.lqws.cn/news/66475.html

相关文章:

  • sigmastar实现SD卡升级
  • 探索大语言模型(LLM):参数量背后的“黄金公式”与Scaling Law的启示
  • 【MATLAB代码】制导——平行接近法,三维,目标是运动的,订阅专栏后可直接查看MATLAB源代码
  • 百度蜘蛛池的作用是什么?技术@baidutopseo
  • deepseek问答记录:请讲解一下transformers.HfArgumentParser()
  • 【Linux】基础文件IO
  • 智慧零工平台后端开发实战:Spring Boot 3 + MyBatis-Flex 现代化架构
  • 【AI论文】表R1:表格推理的推理时间扩展
  • 深入了解linux系统—— 进程间通信之管道
  • 微服务常用日志追踪方案:Sleuth + Zipkin + ELK
  • ffmpeg 的视频格式转换 c# win10
  • 从 AMQP 到 RabbitMQ:核心组件设计与工作原理(二)
  • 【git-首次初始化本地项目、关联远程仓库】
  • 数字规则:进制转换与原码、反码、补码
  • Kotlin 扩展函数详解
  • SQL进阶之旅 Day 12:分组聚合与HAVING高效应用
  • 微软markitdown PDF/WORD/HTML文档转Markdown格式软件整合包下载
  • 智启未来:当知识库遇见莫奈的调色盘——API工作流重构企业服务美学
  • 【MySQL】事务
  • list类的详细讲解
  • JavaScript新特性structuredClone(),一行代码优化深拷贝策略
  • 数学分析——一致性(均匀性)和收敛
  • python打卡day42@浙大疏锦行
  • 湖北理元理律师事务所:个人债务管理的温度与精度
  • 劫持进程注入
  • 子串题解——和为 K 的子数组【LeetCode】
  • 自编码器Auto-encoder(李宏毅)
  • WSL2 安装与Docker安装
  • CP4-OFDM模糊函数原理及仿真
  • HTTPS