百度360必应搜狗淘宝本站头条
当前位置:网站首页 > 技术资源 > 正文

Java设计模式系列之装饰器模式介绍

lipiwang 2024-10-25 15:54 10 浏览 0 评论

一、装饰器模式介绍

1.1 装饰器模式定义

装饰模式(decorator pattern) 的原始定义是:动态的给一个对象添加一些额外的职责。就扩展功能而言,装饰器模式提供了一种比使用子类更加灵活的替代方案。



假设现在有有一块蛋糕,如果只有涂上奶油那这个蛋糕就是普通的奶油蛋糕,这时如果我们添加上一些蓝莓,那这个蛋糕就是蓝莓蛋糕。如果我们再拿一块黑巧克力 然后写上姓名、插上代表年龄的蜡烛,这就是变成了一块生日蛋糕。



在软件设计中,装饰器模式是一种用于替代继承的技术,它通过一种无须定义子类的方式给对象动态的增加职责,使用对象之间的关联关系取代类之间的继承关系。

1.2 装饰器模式原理

1.2.1 模式类图



1.2.2 模式角色说明

装饰(Decorator)模式中的角色:

  • 抽象构件(Component)角色 :它是具体构件和抽象装饰类的共同父类,声明了在具体构件中实现的业务方法。它引进了可以使客户端以一致的方式处理未被装饰的对象以及装饰之后的对象,实现客户端的透明操作。
  • 具体构件(Concrete Component)角色 :它是抽象构件类的子类,用于定义具体的构建对象,实现了在抽象构建中声明的方法,装饰类可以给它增加额外的职责(方法)。
  • 抽象装饰(Decorator)角色 :它也是抽象构件类的子类,用于给具体构件增加职责,但是具体职责在其子类中实现。它维护了一个指向抽象构件对象的引用,通过该引用可以调用装饰之前构件对象的方法,并通过其子类扩展该方法,以达到装饰的目的。
  • 具体装饰(ConcreteDecorator)角色 : 它是抽象装饰类的子类,负责向构件添加新的职责。每一个具体装饰类都定义了一些新的行为,它可以调用在抽象装饰类中定义的方法,并可以增加新的方法用于扩充对象的行为。

1.2.3 举例代码

package main.java.cn.test.decorator.V1;

/**

* @author ningzhaosheng

* @date 2024/1/12 18:46:19

* @description 抽象构件类

*/

public abstract class Component {

//抽象方法

public abstract void operation();

}
package main.java.cn.test.decorator.V1;

/**

* @author ningzhaosheng

* @date 2024/1/12 18:46:54

* @description 具体构建类

*/

public class ConcreteComponent extends Component{

@Override

public void operation() {

//基础功能实现(复杂功能通过装饰类进行扩展)

}

}
package main.java.cn.test.decorator.V1;

/**

* @author ningzhaosheng

* @date 2024/1/12 18:47:38

* @description 抽象装饰类-装饰者模式的核心

*/

public class Decorator extends Component {

//维持一个对抽象构件对象的引用

private Component component;

//注入一个抽象构件类型的对象

public Decorator(Component component) {

this.component = component;

}

@Override

public void operation() {

//调用原有业务方法(这里并没有真正实施装饰,而是提供了一个统一的接口,将装饰过程交给子类完成)

component.operation();

}

}
package main.java.cn.test.decorator.V1;

/**

* @author ningzhaosheng

* @date 2024/1/12 18:48:48

* @description 具体装饰类

*/

public class ConcreteDecorator extends Decorator {

public ConcreteDecorator(Component component) {

super(component);

}

@Override

public void operation() {

super.operation(); //调用原有业务方法

addedBehavior(); //调用新增业务方法

}

//新增业务方法

public void addedBehavior() {

//......

}

}

二、装饰器模式的应用

2.1 需求说明

我们以一个文件读写器程序为例, 演示一下装饰者模式的使用。

2.2 需求实现

2.2.1 类图



2.2.2 类图说明

  • DataLoader

抽象的文件读取接口DataLoader

  • BaseFileDataLoader

具体组件BaseFileDataLoader,重写组件 DataLoader 的读写方法

  • DataLoaderDecorator

装饰器DataLoaderDecorator,这里要包含一个引用 DataLoader 的对象实例 wrapper,同样是重写 DataLoader 方法,不过这里使用wrapper 来读写,并不进行扩展。

  • EncryptionDataDecorator

读写时有加解密功能的具体装饰器EncryptionDataDecorator,它继承了装饰器 DataLoaderDecorator 重写读写方法。

2.2.3 具体实现

2.2.3.1 DataLoader类

package main.java.cn.test.decorator.V2;

/**

* @author ningzhaosheng

* @date 2024/1/12 18:28:43

* @description 抽象的文件读取接口DataLoader

*/

public interface DataLoader {

String read();

void write(String data);

}

2.2.3.2 BaseFileDataLoader类

package main.java.cn.test.decorator.V2;

import java.io.File;

import java.io.IOException;

/**

* @author ningzhaosheng

* @date 2024/1/12 18:29:10

* @description 具体实现组件, 重写读写方法

*/

public class BaseFileDataLoader implements DataLoader {

private String filePath;

public BaseFileDataLoader(String filePath) {

this.filePath = filePath;

}

@Override

public String read() {

try {

String result = FileUtils.readFileToString(new

File(filePath), "utf-8");

return result;

} catch (IOException e) {

e.printStackTrace();

return null;

}

}

@Override

public void write(String data) {

try {

FileUtils.writeStringToFile(new File(filePath),

data, "utf-8");

} catch (IOException e) {

e.printStackTrace();

}

}

}

2.2.3.3 DataLoaderDecorator类

package main.java.cn.test.decorator.V2;

/**

* @author ningzhaosheng

* @date 2024/1/12 18:33:23

* @description 装抽象饰者类

*/

public class DataLoaderDecorator implements DataLoader {

private DataLoader wrapper;

public DataLoaderDecorator(DataLoader wrapper) {

this.wrapper = wrapper;

}

@Override

public String read() {

return wrapper.read();

}

@Override

public void write(String data) {

wrapper.write(data);

}

}

2.2.3.4 EncryptionDataDecorator

package main.java.cn.test.decorator.V2;

import java.util.Base64;

/**

* @author ningzhaosheng

* @date 2024/1/12 18:34:28

* @description 具体装饰者-对文件内容进行加密和解密

*/

public class EncryptionDataDecorator extends DataLoaderDecorator {

public EncryptionDataDecorator(DataLoader wrapper) {

super(wrapper);

}

@Override

public String read() {

return decode(super.read());

}

@Override

public void write(String data) {

super.write(encode(data));

}

//加密操作

private String encode(String data) {

try {

Base64.Encoder encoder = Base64.getEncoder();

byte[] bytes = data.getBytes("UTF-8");

String result = encoder.encodeToString(bytes);

return result;

} catch (Exception e) {

e.printStackTrace();

return null;

}

}

//解密

private String decode(String data) {

try {

Base64.Decoder decoder = Base64.getDecoder();

String result = new String(decoder.decode(data),

"UTF-8");

return result;

} catch (Exception e) {

e.printStackTrace();

return null;

}

}

}

2.2.3.5 TestDecorator测试类

package main.java.cn.test.decorator.V2;

/**

* @author ningzhaosheng

* @date 2024/1/12 18:36:41

* @description 测试类

*/

public class TestDecorator {

public static void main(String[] args) {

String info = "name:tom,age:15";

DataLoaderDecorator decorator = new

EncryptionDataDecorator(new BaseFileDataLoader("demo.txt"));

decorator.write(info);

String data = decorator.read();

System.out.println(data);

}

}

三、装饰器模式总结

3.1 装饰器模式优点

1. 对于扩展一个对象的功能,装饰模式比继承更加灵活,不会导致类的个数急剧增加。

2. 可以通过一种动态的方式来扩展一个对象的功能,通过配置文件可以在运行时选择不同的具体装饰类,从而实现不同的行为。

3. 可以对一个对象进行多次装饰,通过使用不同的具体装饰类以及这些装饰类的排列组合可以创造出很多不同行为的组合,得到更加强大的对象。

4. 具体构建类与具体装饰类可以独立变化,用户可以根据需要增加新的具体构建类和具体装饰类,原有类库代码无序改变,符合开闭原则。

3.2 装饰器模式缺点

1. 在使用装饰模式进行系统设计时将产生很多小对象,这些对象的区别在于它们之间相互连接的方式有所不同,而不是它们的类或者属性值不同,大量的小对象的产生势必会占用更多的系统资源,在一定程度上影响程序的性能。

2. 装饰器模式提供了一种比继承更加灵活、机动的解决方案,但同时也意味着比继承更加易于出错,排错也更加困难,对于多次装饰的对象,在调试寻找错误时可能需要逐级排查,较为烦琐。

3.3 装饰器模式适用场景

1. 快速动态扩展和撤销一个类的功能场景。 比如,有的场景下对 API 接口的安全性要求较高,那么就可以使用装饰模式对传输的字符串数据进行压缩或加密。如果安全性要求不高,则可以不使用。

2. 不支持继承扩展类的场景。 比如,使用 final 关键字的类,或者系统中存在大量通过继承产生的子类。


今天关于Java设计模式系列之装饰器模式介绍的相关内容就分享到这里!

如果对您有帮助,欢迎点赞+关注,也可以发表您宝贵的评论,和我一起互动!


相关推荐

前端 JavaScript 字符串中提取数字

varstr="4500元";varnum=parseInt(str);alert(num);//4500如果字符串前面有非数字字符,上面这种方法就不行了:var...

使用JavaScript如何获取网站网址(js如何获取浏览器信息)

在做网站开发时,我们有时候会获取当前页面的完整路径。在网页前端如何实现呢?请在网页脚本代码段中粘贴如下代码。functiongetRootPath(){//获取当前网址,...

java文本对比工具源码8(java比较文本相似度)

/***ParseatextualrepresentationofpatchesandreturnaListofPatch*objects.*@paramtextline...

JavaScript实现的9大排序算法(js排序方法)

笔试面试经常涉及各种算法,本文简要介绍常用的一些算法,并用JavaScript实现。1、插入排序1)算法简介插入排序(Insertion-Sort)的算法描述是一种简单直观的排序算法。它的工作原理是通...

使用函数化的Javascript代码编写方式

对于Javascript介绍想必大家都耳熟能详啦,基于函数化的编程语言,基于浏览器运行的编程语言,web开发语言,前端开发必备语言,blablabla...Javascript是一个非常灵...

Js基础31:内置对象(js内置对象是什么意思)

js里面的对象分成三大类:内置对象ArrayDateMath宿主对象浏览器提供的对象(如bom、dom等等)自定义对象开发人员自己定义的对象内置对象——所谓内置对象,就是JavaScript自...

js获取上传文件类型以及大小的方法

前端web上传文件时,需要在上传之前判断一下文件的类型以及文件的大小,HTML为前端的标记语言是无法做到这一点,只能使用javascript动态脚本代码来实现。js获取上传文件大小的方法示例代码:&...

黑客入门实践:如何绕过前端过滤上传文件

今天开始,我们就要开始学习具体的漏洞了,但是希望大家学完后,不要轻易"入侵"网站哦,当时测试环境下除外。今天的课程是关于“文件上传漏洞”,据安界网的老师介绍,文件上传漏洞仅次于命令执行...

Java文件上传细讲(java文件夹上传)

什么是文件上传?文件上传就是把用户的信息保存起来。为什么需要文件上传?在用户注册的时候,可能需要用户提交照片。那么这张照片就应该要进行保存。免费学习资料获取方式上传组件(工具)为什么我们要使用上传工具...

鸿蒙上实现“翻译”功能(鸿蒙宴全文翻译)

本章节我们来制作中文翻译成英文的实例(运行在HarmonyOS上),通过HTTP去配合API进行实现。需求分析如下:文字输入HTTP协议使用文字翻译控件介绍①HTTP数据请求官方文档请求...

JavaScript从入门到精通(javascript 入门教程)

前几天,我们学习了JavaScript的入门课程,但是要想做网站,仅仅学会入门是不够的,今后的几天,我将带领大家精通JavaScript,希望大家好好学习!JS内置对象String对象:字符串对象,提...

第15天|16天搞定前端,javascript语法篇(干货)

JavaScript是互联网上最流行的脚本语言,这门语言可用于HTML和web,可广泛用于服务器、PC、笔记本电脑、平板电脑和智能手机等设备。它是一个脚本语言,它是一个轻量级,但功能强大的编程...

通过js来实现打字效果(js如何输入)

有时候浏览网页经常会看见一些页面出现一些打字的效果,那么是怎么实现的呢?逻辑确定目标容器,在哪个容器进行输出确定输出内容,当前直接根据目标容器确定输出内容即可需要控制输出频率,因此需要循环输出完毕代码...

手把手教你学会一键还原混淆js原理

1.短变量名在以下示例代码中,我们将变量“customerName”替换为“a”:vara="JohnSmith";console.log(a);2.随机变量名在以下示例代码...

sql中常用的字符串函数详解(sql字符串函数有哪些)

在日常开发中遇到处理最多的可能字符串要算其中一个了,什么替换啊截取啊大小写转换啊、删除空格啊等等,这些操作我们可以在前端操作,也可以直接在数据库的sql中操作,那么我们来看一下sql中处理字符串的...

取消回复欢迎 发表评论: