news 2026/4/15 14:46:38

Java设计模式——装饰器模式

作者头像

张小明

前端开发工程师

1.2k 24
文章封面图
Java设计模式——装饰器模式

当我们想对一个类进行功能扩展的时候,最简单的方法就是继承该类然后进行修改,但是一个接口下的实现类很多,每个子类都进行继承扩展的话又会诞生很多子类,造成类爆炸的情况。

装饰器模式属于结构型设计模式,就可以做到在不改变原代码的情况下完成功能的扩展,符合OCP原则,也避免了类爆炸的情况。

Java中的IO流大量使用了装饰器模式,可以参考官方的源码进行学习。

  • 在装饰器设计中,两个重要角色:装饰者与被装饰者

  • 装饰器设计模式中,要求装饰者与被装饰者应实现同一个接口/同一些接口/继承同一个抽象类

  • 因为实现同一个接口以后,对于客户端程序来说,使用装饰者时就像在使用被装饰者一样

  • 装饰者中含有被装饰者的引用(A has a B),尽量使用has a[耦合度低一些],不要使用is a(即组合优于继承原则)

例如我们想实现一个自己的Set,能够记录下被移除的元素,有些人可能会想到继承个set,但是我们需要注意,继承只能单继承,这样操作会使该类失去其他继承的资格,所以优先还是实现set接口而不影响原本的继承扩展。

-HistorySet.java

packageinsight;importjava.util.*;/** * @author: wangcai * @date: 2025/12/15 */publicclassHistorySet<E>implementsSet<E>{/* 记录Remove掉的元素 */List<E>removeList=newArrayList<>();finalSet<E>delegate;publicHistorySet(Set<E>delegate){this.delegate=delegate;}@Overridepublicintsize(){returndelegate.size();}@OverridepublicbooleanisEmpty(){returndelegate.isEmpty();}@Overridepublicbooleancontains(Objecto){returndelegate.contains(o);}@OverridepublicIteratoriterator(){returndelegate.iterator();}@OverridepublicObject[]toArray(){returnnewObject[0];}@Overridepublicbooleanadd(Objecto){returndelegate.add((E)o);}@Overridepublicbooleanremove(Objecto){booleanremove=delegate.remove(o);if(remove){removeList.add((E)o);}returnremove;}@OverridepublicbooleanaddAll(Collectionc){returnfalse;}@Overridepublicvoidclear(){}@OverridepublicbooleanremoveAll(Collectionc){returnfalse;}@OverridepublicbooleanretainAll(Collectionc){returnfalse;}@OverridepublicbooleancontainsAll(Collectionc){returnfalse;}@OverridepublicObject[]toArray(Object[]a){returnnewObject[0];}@OverridepublicStringtoString(){return"HistorySet{ "+"delegate = "+delegate.toString()+" "+"removeList = "+removeList.toString()+" }";}}

例如上面这样,这样可以自己无需实现set的功能就可以具备set的能力。而且支持手动传入set实例,由用户自己控制想要扩展的set,HashSet或者TreeSet。而如果通过继承实现该扩展功能的话,就不能这么灵活,我们需要每种实现都单独编写一个扩展类。

-Usage.java

packageinsight;importjava.util.HashSet;/** * @author: wangcai * @date: 2025/12/15 */publicclassUsage{publicstaticvoidmain(String[]args){HistorySet<Integer>integers=newHistorySet<Integer>(newHashSet<>());integers.add(1);integers.add(2);integers.add(3);integers.add(4);integers.add(5);integers.remove(2);integers.remove(2);integers.remove(5);System.out.println(integers);}}/* HistorySet{ delegate = [1, 3, 4] removeList = [2, 5] } */

并且我们可以反复的包,因为我们实现的本身也是一个set集合,还是能作为构造器参数传入。

packageinsight;importjava.util.HashSet;/** * @author: wangcai * @date: 2025/12/15 */publicclassUsage{publicstaticvoidmain(String[]args){HistorySet<Integer>integers=newHistorySet<>(newHashSet<>());HistorySet<Integer>integers1=newHistorySet<>(integers);integers1.add(1);integers1.add(2);integers1.add(3);integers1.add(4);integers1.add(5);integers1.remove(2);integers1.remove(2);integers1.remove(5);System.out.println(integers1);}}/* HistorySet{ delegate = HistorySet{ delegate = [1, 3, 4] removeList = [2, 5] } removeList = [2, 5] } */

我们可以看一下Java中的一些实现:

Collections.synchronizedCollection()

publicstatic<T>Collection<T>synchronizedCollection(Collection<T>c){returnnewSynchronizedCollection<>(c);}

===BufferedInputStream

我们自己手写一个BufferedInputStream模仿Java IO流中的缓冲输入流来提升读取效率。

-Usage.java

packageinsight.input;importjava.io.File;importjava.io.FileInputStream;importjava.time.Instant;/** * @author: wangcai * @date: 2025/12/16 */publicclassUsage{publicstaticvoidmain(String[]args)throwsException{Filefile=newFile("src/main/resources/深入理解Java虚拟机(第3版).pdf");longmillisecond=Instant.now().toEpochMilli();try(FileInputStreamfileInputStream=newFileInputStream(file)){while(true){intread=fileInputStream.read();if(read==-1){break;}}}System.out.println("耗时: "+(Instant.now().toEpochMilli()-millisecond)+"ms");}}/* 耗时: 62638ms */

可以看的出来,耗时很久,基于此我们利用装饰器模式自己改造一下FileInputStream。缓冲区的核心在于我们需要读取的时候保证效率高,但是又不需要一口气读完,所以我们需要创建一个缓冲区读取文件了保存下缓存内容,需要的时候在内存中直接拿取。

-BufferedFileInputStream.java

packageinsight.input;importjava.io.IOException;importjava.io.InputStream;/** * @author: wangcai * @date: 2025/12/16 */publicclassBufferedFileInputStreamextendsInputStream{privatefinalInputStreamin;// 缓冲区 默认8kbbyte[]buffer=newbyte[8192];// 标志位 为-1时代表不可读 其余为读取的开始位置privateintbufferPos=-1;// 用来避免超出的读取文件privateintcapacity=-1;publicBufferedFileInputStream(InputStreaminputStream){this.in=inputStream;}@Overridepublicintread()throwsIOException{if(buffCanRead()){returnreadFromBuffer();}refreshBuffer();if(!buffCanRead()){return-1;}returnreadFromBuffer();}privateintreadFromBuffer(){returnbuffer[bufferPos++]&0xff;}privatevoidrefreshBuffer()throwsIOException{capacity=this.in.read(buffer);bufferPos=0;}privatebooleanbuffCanRead(){if(capacity==-1){returnfalse;}returnbufferPos!=capacity;}@Overridepublicvoidclose()throwsIOException{super.close();}}
packageinsight.input;importjava.io.File;importjava.io.FileInputStream;importjava.io.InputStream;importjava.time.Instant;/** * @author: wangcai * @date: 2025/12/16 */publicclassUsage{publicstaticvoidmain(String[]args)throwsException{Filefile=newFile("src/main/resources/深入理解Java虚拟机(第3版).pdf");longmillisecond=Instant.now().toEpochMilli();try(InputStreamfileInputStream=newBufferedFileInputStream(newFileInputStream(file))){while(true){intread=fileInputStream.read();if(read==-1){break;}}}System.out.println("耗时: "+(Instant.now().toEpochMilli()-millisecond)+"ms");}}/* 耗时: 49ms */

效率大大提升,这也是官方源码中类似的用法的实践。

版权声明: 本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若内容造成侵权/违法违规/事实不符,请联系邮箱:809451989@qq.com进行投诉反馈,一经查实,立即删除!
网站建设 2026/4/12 14:44:49

环形数组+位运算+双向链表:手把手教你实现一个生产级C++定时器系统

在现代软件系统中,定时器是不可或缺的基础设施。无论是游戏服务器的心跳检测、网络请求的超时控制,还是任务调度系统的定时执行,都离不开高效、精确的定时器机制。然而,设计一个在各种时间尺度下(从毫秒到天)都能保持高性能的定时器系统并非易事。传统的基于最小堆的定时…

作者头像 李华
网站建设 2026/4/11 22:08:06

【运维自动化-标准运维】如何创建条件分支流程

什么是分支网关 分支网关&#xff1a;用来标识分支流程的开始&#xff0c;通过和汇聚网关配对使用&#xff0c;可以根据分支表达式的计算结果&#xff0c;动态的控制多个分支中某一个分支执行。 如何使用 1、编辑流程配置第一个节点配置 执行失败、执行成功 节点配置分支条件执…

作者头像 李华
网站建设 2026/4/7 5:53:18

为什么顶尖量子工程师都在用VSCode跑Jupyter?,真相就在这6个片段中

第一章&#xff1a;为什么VSCode Jupyter 成为量子计算开发新标准在量子计算快速发展的今天&#xff0c;开发环境的易用性与集成能力成为推动研究和应用落地的关键。VSCode 与 Jupyter 的深度整合&#xff0c;正逐步确立其作为该领域新标准的地位。无缝的交互式编程体验 Jupyt…

作者头像 李华