设计模式之装饰者模式(三)

欢迎大家的持续关注。上一次,我们结合第一篇推导出来的类图,到第二篇根据类图进行实际代码的编写,对装饰者模式有了一个整体的概念以及实战。不知道对你帮助如何呢?小编已经有门道了,看完接下来的一部分,你会恍然大悟,原来实际编码中你一直在用装饰者模式。

真实世界的装饰者:Java I/O

看到标题,是不是就很想往下看,到底是I/O中的什么呢,让你早已经拥有了装饰者模式的实践?就如书上给的描述,你第一次(还有第二次和第三次)看到这些API发出“哇”的惊叹时,放心,你不是唯一收到惊吓的人。下面,我们马上给出一个典型的对象集合,用装饰这来将功能结合起来,以读取文件数据:

装饰 java.io 类


你发现了没,java.io和之前的咖啡店其实没有太大的差异。这里使用了各种“输入”流装饰者来符合你的用途。
你会发现“输出”流的设计方式也是一样的。你还会发现Reader/Writer流和输入流/输出流的类相当类似。不过,出学过Java文件流的同学肯定知道,Java I/O也引出类装饰者的一个“缺点”:利用装饰者模式,常常造成设计中有大量的小类,数量实在太多,可能会造成使用此API程序员的困扰。不知道你之前有没有搞晕过呢,小编初入Java I/O的世界的时候,很是晕头转向。

编写自己的Java I/O 装饰者

那么,已经懂得了装饰者的精髓,我们自己来编写一个输入的装饰者如何呢?来一个小想法:编写一个装饰者,把输入流内的所有大写字符转成小写。赶紧行动吧。

import java.io.FilterInputStream;
import java.io.IOException;
import java.io.InputStream;

public class LowerCaseInputStream extends FilterInputStream {

    protected LowerCaseInputStream(InputStream in) {
        super(in);
    }

    public int read() throws IOException {
        int c = in.read();
        return (c == -1 ? c : Character.toLowerCase((char)c));
    }

    public int read(byte[] b,int offset,int len) throws IOException {
        int result = in.read(b, offset, len);
        for (int i = offset; i < offset+result; i++) {
            b[i] = (byte)Character.toLowerCase((char)b[i]);
        }
        return result;
    }

}

测试你的新Java I/O 装饰者

基于上面的装饰者,我们来实现下功能,看是否是符合你的需求呢?

public class InputTest {

    public static void main(String[] args) throws IOException{
        int c;
        try {
            // 设置FileInoutStream,先用BufferedInputStream装饰它,再用我们崭新的LowerCaseInputStream过滤器装饰它
            InputStream inputStream = new LowerCaseInputStream(
                    new BufferedInputStream(
                            new FileInputStream("test.txt")));

            while ((c = inputStream.read()) >= 0) {
                System.out.print((char)c);
            }
            inputStream.close();
        } catch (IOException e) {

        }
    }
}

// 输入内容:
I know the Decorator Pattern and HOW it's used in the JAVA.IO package.
// 运行结果
i know the decorator pattern and how it's used in the java.io package.

恭喜你,已经喜提装饰者模式,在设计模式的世界里又更进一步了。这里,我们通过咖啡厅卖升级订单系统,来了解装饰者模式一步步产生的过程;再通过实际的Java API中的I/O进一步巩固他,并用自定义的I/O功能实现,来结束此次的装饰者模式之旅,开不开心呢?

设计箱内的工具

还是按照之前的套路,总结下工具箱内新增的工具吧

  • OO基础 本次没有变化

    抽象、封装、继承、多态

  • OO原则

    封装变化

    多用组合,少用继承

    针对接口编程,不针对实现编程

    为交互对象之间的松耦合设计而努力

    对扩展开放,对修改关闭(现在有了开放-关闭原则引导我们。我们会努力的设计系统,好让关闭的部分和新扩展的部分隔离)

  • OO模式

    『策略模式』

    『观察者模式』

    『装饰者模式』动态地将责任附加到对象上。想要扩展功能,装饰者提供有别于继承的另一种选择

遗留问题解答

哦,对了,上次还留下了一个新增的功能,在咖啡厅里顾客可以选择小杯、中杯、大杯的需求,调料根据咖啡容量收费。例如:小中大杯的咖啡加上豆浆,分别加收0.10,、0.15、0.20。那怎么做呢,请往下看。这里仅列举Soy类,其他类的代码请移步到GitHub上查看。

public class Soy extends CondimentDecorator {

    Beverage beverage;

    public Soy(Beverage beverage) {
        this.beverage = beverage;
    }

    @Override
    public String getDescription() {
        return beverage.getDescription() + " , Soy";
    }

    @Override
    public double cost() {
        double cost = beverage.cost();
        if (beverage.getSize() == Size.TALL) {
            cost += 0.10;
        } else if (beverage.getSize() == Size.GRANDE) {
            cost += 0.15;
        } else if (beverage.getSize() == Size.VENTI) {
            cost += 0.20;
        }
        return cost;
    }

}

至此,我们又学完了一个设计模式,装饰者模式。除了之前的要点之外,小编在这里还得提醒下大家,装饰者一般对组件的客户是透明的,除非客户程序依赖于组件的具体类型。你可以用无数个装饰者包装一个组件,但是这样会导致设计中出现许多小对象,如果过度使用,会让程序变得很复杂。所以,我们需要按需使用,而不是滥用噢。

已经陆陆续续学完了3个设计模式,不知道大家对这样的方式有没有不解,或者有没有更多的建议,小编都会虚心接受。下次课程,我们学习工厂模式。

PS:代码已经上传,需要查看的朋友点击此处HeadFirstDesign

推荐阅读

设计模式之装饰者模式(一)

设计模式之装饰者模式(二)

爱生活,爱学习,爱感悟,爱挨踢

原文地址:https://www.cnblogs.com/dimple91/p/10700688.html

时间: 2024-12-17 17:11:58

设计模式之装饰者模式(三)的相关文章

设计模式 2 —— 装饰者模式

设计模式目录: 设计模式 1 ——观察者模式 设计模式 2 —— 装饰者模式

Java经典23种设计模式之行为型模式(三)

本文接着介绍11种行为型模式里的备忘录模式.观察者模式.状态模式. 一.备忘录模式 在不破坏封装性的前提下,捕获一个对象的内部状态,并在该对象之外保存这个状态.这样以后就可以将该对象恢复到原先保存的状态.还是比较好理解的. 1.Memento 备忘录存储原发器对象的内部状态,这个类就是要存储的对象的状态.状态需要多少个变量,在Memento里就写多少个变量. public class Memento { private String state; public Meme*to(String st

Java经典23种设计模式之结构型模式(三)------附代理模式、适配器模式、外观模式区别

本文介绍7种结构型模式里的剩下两种:享元模式.代理模式. 一.享元模式FlyWeight 享元模式比较简单且重要,在很多场合都被用到,只不过封装起来了用户看不到.其概念:运用共享内存技术最大限度的支持大量细粒度的对象.这个概念给的有些抽象,说白了就是如果内存中存在某个对象A,如果再次需要使用对象A的时候如果内存中有A这个对象就直接使用它,不要再次new了.如果没有,则重新new一个.基于这个特点,享元模式使用时一般会给待访问对象传递一个Tag,用来标识这个对象,而且要同时使用抽象工厂的方法进行访

Design Pattern 设计模式 Decorator 装饰者模式

1 设计一个基类 2 继承这个基类,并包含这个基类的一个对象 3 创建继承类的时候,初始化这个基类,因为是基类指针,所以可以是所有基类的衍生类对象,那么就可以叠加任意多个衍生类对象了. 关键是在衍生类中包含一个基类对象,然后有了继承和包含两重关系,可以使得一个类一个类对象叠加,达到装饰目的. 等于是创建了一个对象指针链表,一个一个对象发挥其相应的作用. 下面程序装饰一个桌子,同时包含释放内存的处理. 关键要知道基类的析构函数必须使用虚函数,为什么? 1 如果基类不是析构函数,那么如果衍生类中包含

java设计模式之 装饰器模式

适AT java设计模式之 装饰器模式 装饰器模式 装饰器模式(Decorator Pattern)允许向一个现有的对象添加新的功能,同时又不改变其结构. 这种类型的设计模式属于结构型模式,它是作为现有的类的一个包装. 这种模式创建了一个装饰类,用来包装原有的类,并在保持类方法签名完整性的前提下,动态给一个对象添提供了额外的功能. 我们通过下面的实例来演示装饰器模式的用法.模拟一个人从想吃饭.找饭店.享受美食.结束吃饭的过程 代码展示: 首先创建一个被修饰的接口 Eat package deco

设计模式之装饰者模式(Decorator)

1.定义 装饰者模式动态地将责任附加到对象上.若要扩展功能,装饰者提供了比继承更有弹性的替代方案 2.类图 Component抽象构件:Component是一个接口或者是抽象类,就是定义我们最核心的对象,也就是最原始的对象.在装饰者模式中,必然有一个最基本.最核心.最原始的接口或抽象类充当Component抽象构件. ConcreteComponent具体构件:ConcreteComponent是最核心.最原始.最基本的接口或抽象类的实现,你要装饰的就是它. Decorator装饰角色:一般是一

掌握设计模式之装饰者模式

![](http://ww4.sinaimg.cn/large/006tNc79ly1g4ztauvhzej30p00dw45p.jpg) ## 前言 当应用开发中,我们要为一个对象在原有功能上进行扩展增强时,往往采用继承的方式,而继承过多时就会使得功能类更加复杂,不利于维护,而设计模式中装饰者模式可以帮助我们更好对应这种场景,装饰者模式可以做到让对象能够动态地进行功能扩展,而不影响其他对象. 那究竟它是如何实现的呢,又如何实际应用呢,就让我们一起来学习下这个模式吧. ## 模式定义 装饰者模式

java_设计模式_装饰者模式_Decorator Pattern(2016-07-28)

装饰模式又名包装(Wrapper)模式.装饰模式以对客户端透明的方式扩展对象的功能,是继承关系的一个替代方案. 装饰模式的结构 装饰模式以对客户透明的方式动态地给一个对象附加上更多的责任.换言之,客户端并不会觉得对象在装饰前和装饰后有什么不同.装饰模式可以在不使用创造更多子类的情况下,将对象的功能加以扩展. 装饰模式的类图如下: 在装饰模式中的角色有: ● 抽象构件(Component)角色:给出一个抽象接口,以规范准备接收附加责任的对象. ● 具体构件(ConcreteComponent)角色

设计模式之装饰器模式

一.百科 概述: 23种设计模式之一,英文叫Decorator Pattern,又叫装饰者模式.装饰模式是在不必改变原类文件和使用继承的情况下,动态地扩展一个对象的功能.它是通过创建一个包装对象,也就是装饰来包裹真实的对象. 特点: (1) 装饰对象和真实对象有相同的接口.这样客户端对象就能以和真实对象相同的方式和装饰对象交互. (2) 装饰对象包含一个真实对象的引用(reference) (3) 装饰对象接受所有来自客户端的请求.它把这些请求转发给真实的对象. (4) 装饰对象可以在转发这些请