news 2026/4/18 7:09:23

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/17 18:21:40

Excel Ctrl+F 平替神器,小白躺平查数据(绿色版直装)

谁懂啊家人们&#xff01;明明只是想查个表格数据&#xff0c;却要被迫打开好几百兆的 Excel&#xff0c;等加载半天不说&#xff0c;CtrlF 搜完还得在密密麻麻的单元格里找结果&#xff0c;懒人真的会谢&#xff01; 直到我挖到这款Excel 数据查询绿色工具&#xff0c;直接把…

作者头像 李华
网站建设 2026/4/18 8:42:01

从 G1 到 G7:台式扫描电镜在稳定性与自动化上的技术演进

自 2006 年第一代 Phenom G1 台式扫描电镜推出以来&#xff0c;该系列已完成 7 代技术演进。2025 年&#xff0c;第七代旗舰 Phenom P 系列 G7&#xff08;Phenom ProX G7、Phenom Pro G7、Phenom Pure G7&#xff09;正式发布。 19 年产品迭代的核心目标始终明确&#xff1a;…

作者头像 李华
网站建设 2026/4/18 8:42:11

智能内容本地化革命:打造永久收藏的数字宝库

在数字阅读日益普及的今天&#xff0c;你是否曾因网络中断而无法继续阅读心爱的电子书&#xff1f;是否在旅途中因信号问题而错失宝贵的阅读时光&#xff1f;智能内容本地化工具正是为解决这些痛点而生&#xff0c;它将帮助你将云端阅读体验转化为永久收藏。 【免费下载链接】s…

作者头像 李华
网站建设 2026/4/18 8:20:21

【开题答辩全过程】以 基于SSM的考研信息共享平台为例,包含答辩的问题和答案

个人简介一名14年经验的资深毕设内行人&#xff0c;语言擅长Java、php、微信小程序、Python、Golang、安卓Android等开发项目包括大数据、深度学习、网站、小程序、安卓、算法。平常会做一些项目定制化开发、代码讲解、答辩教学、文档编写、也懂一些降重方面的技巧。感谢大家的…

作者头像 李华
网站建设 2026/4/17 8:56:06

这才是SpirngBoot轻松整合MybatisPlus多数据源的正确打开方式

有一个这样子的需求&#xff0c;线上正在跑的业务&#xff0c;由于业务发展需要&#xff0c;需重新开发一套新系统&#xff0c;等新系统开发完成后&#xff0c;需要无缝对接切换&#xff0c;当初具体设计见草图。添加依赖#<!--lombok--><dependency><groupId>…

作者头像 李华