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

相关文章

HarmonyOS学习(七)——UI(五)常用布局总结

自适应布局 1.1、线性布局(LinearLayout) 通过线性容器Row和Column实现线性布局。Column容器内的子组件按照垂直方向排列,Row组件中的子组件按照水平方向排列。 属性说明space通过space参数设置主轴上子组件的间距,达到各子组件在排列上的等间距效果alignItems设置子组件在交叉轴上的对齐方式,且在各类尺寸屏幕上表现一致,其中交叉轴为垂直时,取值为Vert

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

Spring Security 基于表达式的权限控制

前言 spring security 3.0已经可以使用spring el表达式来控制授权,允许在表达式中使用复杂的布尔逻辑来控制访问的权限。 常见的表达式 Spring Security可用表达式对象的基类是SecurityExpressionRoot。 表达式描述hasRole([role])用户拥有制定的角色时返回true (Spring security默认会带有ROLE_前缀),去

浅析Spring Security认证过程

类图 为了方便理解Spring Security认证流程,特意画了如下的类图,包含相关的核心认证类 概述 核心验证器 AuthenticationManager 该对象提供了认证方法的入口,接收一个Authentiaton对象作为参数; public interface AuthenticationManager {Authentication authenticate(Authenti

Spring Security--Architecture Overview

1 核心组件 这一节主要介绍一些在Spring Security中常见且核心的Java类,它们之间的依赖,构建起了整个框架。想要理解整个架构,最起码得对这些类眼熟。 1.1 SecurityContextHolder SecurityContextHolder用于存储安全上下文(security context)的信息。当前操作的用户是谁,该用户是否已经被认证,他拥有哪些角色权限…这些都被保

Spring Security基于数据库验证流程详解

Spring Security 校验流程图 相关解释说明(认真看哦) AbstractAuthenticationProcessingFilter 抽象类 /*** 调用 #requiresAuthentication(HttpServletRequest, HttpServletResponse) 决定是否需要进行验证操作。* 如果需要验证,则会调用 #attemptAuthentica

Spring Security 从入门到进阶系列教程

Spring Security 入门系列 《保护 Web 应用的安全》 《Spring-Security-入门(一):登录与退出》 《Spring-Security-入门(二):基于数据库验证》 《Spring-Security-入门(三):密码加密》 《Spring-Security-入门(四):自定义-Filter》 《Spring-Security-入门(五):在 Sprin

Java架构师知识体认识

源码分析 常用设计模式 Proxy代理模式Factory工厂模式Singleton单例模式Delegate委派模式Strategy策略模式Prototype原型模式Template模板模式 Spring5 beans 接口实例化代理Bean操作 Context Ioc容器设计原理及高级特性Aop设计原理Factorybean与Beanfactory Transaction 声明式事物

Java进阶13讲__第12讲_1/2

多线程、线程池 1.  线程概念 1.1  什么是线程 1.2  线程的好处 2.   创建线程的三种方式 注意事项 2.1  继承Thread类 2.1.1 认识  2.1.2  编码实现  package cn.hdc.oop10.Thread;import org.slf4j.Logger;import org.slf4j.LoggerFactory

学习hash总结

2014/1/29/   最近刚开始学hash,名字很陌生,但是hash的思想却很熟悉,以前早就做过此类的题,但是不知道这就是hash思想而已,说白了hash就是一个映射,往往灵活利用数组的下标来实现算法,hash的作用:1、判重;2、统计次数;