Java IO 装饰模式(Decorator)”总结

2024-05-27 10:32

本文主要是介绍Java IO 装饰模式(Decorator)”总结,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

[color=red]JAVA里面用到的最重要的一个设计模式:“装饰模式(Decorator)”。几乎IO整个体系里面都用到这个模式[/color]。

• 装饰模式又名包装(Wrapper)模式。
• 装饰模式以对客户端透明的方式扩展[color=red]对象[/color]的功能,是继承关系的一个替代方案。【例如继承是用来扩展类的功能的,父类定义了一些方法,子类继承这些方法,那么子类的功能就扩展了。而这个模式是用来扩展对象的功能的,对象之间互相组合就能完成不同的功能。】
• 装饰模式以对客户透明的方式[color=red]动态[/color]的给一个对象附加上更多的责任。换言之,客户端并不会觉得对象在装饰前和装饰后有什么不同。
• [color=red]装饰模式可以在不创造更多子类的情况下,将对象的功能加以扩展[/color]。
• 装饰模式[color=red]把客户端的调用委派到被装饰类[/color]。装饰模式的关键在于这种扩展完全是透明
的。
• 装饰模式是在不必改变原类文件和使用继承的情况下,动态的扩展一个对象的功能。它是通过创建一个包装对象,也就是装饰来包裹真实的对象。


DataOutputStream dos = new DataOutputStream(new BufferedOutputStream(
new FileOutputStream("data.txt")));
• 装饰模式的角色:
–[color=red]抽象构件角色(Component)[/color]:给出一个抽象接口,以规范准备接收附加责任的对象。【例如以上代码里面的OutputStream是抽象构建角色】
–[color=red]具体构件角色(Concrete Component)[/color]:定义一个将要接收附加责任的类。【例如以上代码里面的FileOutputStream是一个具体构建角色】
–[color=red]装饰角色(Decorator)[/color]:[color=red]持有一个构件(Component)对象的引用[/color],并定义一个与抽象构件接口一致的接口。【例如以上代码里面的FilterOutputStream是装饰角色】
–[color=red]具体装饰角色(Concrete Decorator)[/color]:负责给构件对象“贴上”附加的责任。【例如以上代码里面的FilterOutputStream类的子类BufferedOutputStream是具体装饰角色】


• 装饰模式的特点:
–[color=red]装饰对象和真实对象有相同的接口。这样客户端对象就可以以和真实对象相同的方式和装饰对象交互[/color]。
–[color=red]装饰对象包含一个真实对象的引用(reference)[/color]
–装饰对象接收所有来自客户端的请求。它把这些请求转发给真实的对象。
–[color=red]装饰对象可以在转发这些请求以前或以后增加一些附加功能[/color]。这样就确保了在运行时,不用修改给定对象的结构就可以在外部增加附加的功能。在面向对象的设计中,通常是通过继承来实现对给定类的功能扩展。


下面来比较装饰模式和继承这两个的特点:
• [color=red]装饰模式[/color]:
–用来扩展特定对象的功能
–不需要子类
–动态
–运行时分配职责
–防止由于子类而导致的复杂和混乱
–更多的灵活性
–对于一个给定的对象,同时可能有不同的装饰对象,客户端可以通过它的需要选择合适的装饰对象发送消息。
• [color=red]继承[/color]:
–用来扩展一类对象的功能
–需要子类
–静态
–编译时分派职责
–导致很多子类产生
–缺乏灵活性


下面是代码实现的一个简单的装饰模式的示例,简单单足以说明问题:
package com.shengshiyuan.decorator;

/**
* 抽象构建角色
* 类: Component <br>
* 描述: TODO <br>
* 作者: fangguanhong fangguanhong@163.com <br>
* 时间: Nov 18, 2013 12:31:27 PM
*/
public interface Component {

public void doSomething();
}

package com.shengshiyuan.decorator;

/**
* 具体构件角色
* 类: ConcreteComponent <br>
* 描述: TODO <br>
* 作者: fangguanhong fangguanhong@163.com <br>
* 时间: Nov 18, 2013 12:31:40 PM
*/
public class ConcreteComponent implements Component {

public void doSomething() {
System.out.println("功能A");
}
}

package com.shengshiyuan.decorator;

/**
* 装饰角色
* 类: Decorator <br>
* 描述: TODO <br>
* 作者: fangguanhong fangguanhong@163.com <br>
* 时间: Nov 18, 2013 12:20:54 PM
*/
public class Decorator implements Component {

private Component component;

public Decorator(Component component) {
this.component = component;
}

public void doSomething() {
component.doSomething();
}
}

package com.shengshiyuan.decorator;

/**
* 具体装饰角色
* 类: ConcreteDecorator1 <br>
* 描述: TODO <br>
* 作者: fangguanhong fangguanhong@163.com <br>
* 时间: Nov 18, 2013 12:28:48 PM
*/
public class ConcreteDecorator1 extends Decorator {

// 父类没有不带参数的构造方法,默认取找找不到,必须手动指明调用父类的哪个构造方法。
public ConcreteDecorator1(Component component) {
super(component);
}

@Override
public void doSomething() {
// 父类完成它的那个主要的功能,子类在下面增加它想要增加的功能。
super.doSomething();

// 子类完成它想要增加的功能。
this.doAnotherThing();
}

private void doAnotherThing() {
System.out.println("功能B");
}
}

package com.shengshiyuan.decorator;

/**
* 具体装饰角色
* 类: ConcreteDecorator2 <br>
* 描述: TODO <br>
* 作者: fangguanhong fangguanhong@163.com <br>
* 时间: Nov 18, 2013 12:31:05 PM
*/
public class ConcreteDecorator2 extends Decorator {

public ConcreteDecorator2(Component component) {
super(component);
}

@Override
public void doSomething() {
super.doSomething();

this.doAnotherThing();
}

private void doAnotherThing() {
System.out.println("功能C");
}
}

package com.shengshiyuan.decorator;

/**
* 测试类
* 类: Client <br>
* 描述: TODO <br>
* 作者: fangguanhong fangguanhong@163.com <br>
* 时间: Nov 18, 2013 2:07:43 PM
*/
public class Client {
public static void main(String[] args) {
// 节点流
Component component = new ConcreteComponent();
// 过滤流
Component component2 = new ConcreteDecorator1(component);
// 过滤流
Component component3 = new ConcreteDecorator2(component2);

component3.doSomething();

System.out.println("===============================================");

// 下面是整合之后的代码
Component comp = new ConcreteDecorator1(new ConcreteDecorator2(
new ConcreteComponent()));
comp.doSomething();
}
}

这篇关于Java IO 装饰模式(Decorator)”总结的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



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

相关文章

Java中流式并行操作parallelStream的原理和使用方法

《Java中流式并行操作parallelStream的原理和使用方法》本文详细介绍了Java中的并行流(parallelStream)的原理、正确使用方法以及在实际业务中的应用案例,并指出在使用并行流... 目录Java中流式并行操作parallelStream0. 问题的产生1. 什么是parallelS

Java中Redisson 的原理深度解析

《Java中Redisson的原理深度解析》Redisson是一个高性能的Redis客户端,它通过将Redis数据结构映射为Java对象和分布式对象,实现了在Java应用中方便地使用Redis,本文... 目录前言一、核心设计理念二、核心架构与通信层1. 基于 Netty 的异步非阻塞通信2. 编解码器三、

SpringBoot基于注解实现数据库字段回填的完整方案

《SpringBoot基于注解实现数据库字段回填的完整方案》这篇文章主要为大家详细介绍了SpringBoot如何基于注解实现数据库字段回填的相关方法,文中的示例代码讲解详细,感兴趣的小伙伴可以了解... 目录数据库表pom.XMLRelationFieldRelationFieldMapping基础的一些代

一篇文章彻底搞懂macOS如何决定java环境

《一篇文章彻底搞懂macOS如何决定java环境》MacOS作为一个功能强大的操作系统,为开发者提供了丰富的开发工具和框架,下面:本文主要介绍macOS如何决定java环境的相关资料,文中通过代码... 目录方法一:使用 which命令方法二:使用 Java_home工具(Apple 官方推荐)那问题来了,

Java HashMap的底层实现原理深度解析

《JavaHashMap的底层实现原理深度解析》HashMap基于数组+链表+红黑树结构,通过哈希算法和扩容机制优化性能,负载因子与树化阈值平衡效率,是Java开发必备的高效数据结构,本文给大家介绍... 目录一、概述:HashMap的宏观结构二、核心数据结构解析1. 数组(桶数组)2. 链表节点(Node

Java AOP面向切面编程的概念和实现方式

《JavaAOP面向切面编程的概念和实现方式》AOP是面向切面编程,通过动态代理将横切关注点(如日志、事务)与核心业务逻辑分离,提升代码复用性和可维护性,本文给大家介绍JavaAOP面向切面编程的概... 目录一、AOP 是什么?二、AOP 的核心概念与实现方式核心概念实现方式三、Spring AOP 的关

详解SpringBoot+Ehcache使用示例

《详解SpringBoot+Ehcache使用示例》本文介绍了SpringBoot中配置Ehcache、自定义get/set方式,并实际使用缓存的过程,文中通过示例代码介绍的非常详细,对大家的学习或者... 目录摘要概念内存与磁盘持久化存储:配置灵活性:编码示例引入依赖:配置ehcache.XML文件:配置

Java 虚拟线程的创建与使用深度解析

《Java虚拟线程的创建与使用深度解析》虚拟线程是Java19中以预览特性形式引入,Java21起正式发布的轻量级线程,本文给大家介绍Java虚拟线程的创建与使用,感兴趣的朋友一起看看吧... 目录一、虚拟线程简介1.1 什么是虚拟线程?1.2 为什么需要虚拟线程?二、虚拟线程与平台线程对比代码对比示例:三

Python版本与package版本兼容性检查方法总结

《Python版本与package版本兼容性检查方法总结》:本文主要介绍Python版本与package版本兼容性检查方法的相关资料,文中提供四种检查方法,分别是pip查询、conda管理、PyP... 目录引言为什么会出现兼容性问题方法一:用 pip 官方命令查询可用版本方法二:conda 管理包环境方法

Java中的.close()举例详解

《Java中的.close()举例详解》.close()方法只适用于通过window.open()打开的弹出窗口,对于浏览器的主窗口,如果没有得到用户允许是不能关闭的,:本文主要介绍Java中的.... 目录当你遇到以下三种情况时,一定要记得使用 .close():用法作用举例如何判断代码中的 input