美文网首页
装饰者模式

装饰者模式

作者: 柚子过来 | 来源:发表于2018-02-06 11:55 被阅读0次

作用:用于动态的扩展已有对象的功能,而不用修改原代码,有别于继承,通过继承来扩展行为时所有子类都会继承到相同的行为,且这种行为是编译时静态决定的,不能动态扩展。

OO原则:

组合优于继承
开放关闭原则:对扩展开放,对修改关闭



Example:
装饰者模式其实就是在需要扩展组件的功能时,抛弃继承的方法。使用装饰者封装新行为,然后将组件作为实例变量封装进装饰者中。一般设计时会有四个部分:抽象组件(或接口)、具体组件、抽象装饰着(或接口)、具体装饰者,抽象装饰者最好实现自抽象组件,因为装饰者与被装饰者必须有相同的类型。
在Java中的I/O实现就是装饰者模式:

wwwwwwwwww.png
上面说了抽象装饰器是为了装饰者与被装饰者类型匹配,这样装饰者装饰后返回的还是原类型,就可以无限多层的装饰被装饰者了。

例如我现在需要扩展InputStream的功能:读到字符后将字符装换成大写
第一步:编写自己的装饰者:

public class UpperCaseInputStream extends FilterInputStream{

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

public int read(byte b[], int off, int len) throws IOException {
    int result = super.read(b,off,len);
    for(int i =0;i<b.length;i++) {
        b[i] = (byte) Character.toUpperCase((char) b[i]);
    }
    return result;
}
}

装饰者将扩展了InputStream的功能,在使用时我们就可以使用包含了新功能的InputStream了:

public class InputTest {
public static void main(String[] args) {
    try {
        byte[] c  = new byte[10];
        InputStream inputStream = new UpperCaseInputStream(new FileInputStream("/qwer.txt"));
        inputStream.read(c,0,10);
        for (byte c1: c) {
            System.out.print((char) c1);
        }
    } catch (Exception e) {
        e.printStackTrace();
    }
}
}
注意该模式与继承的区别:

使用装饰者模式时我们就可以在任何时候通过实现新的装饰者来增加新的行为,但是如果使用继承,当需要新的行为时就需要修改现有的代码。这是组合优于继承、开放关闭的思想,比如上面的UpperCaseInputStream,如果现在有需求说要在将字符全部大写的基础上再将字符全部+1呢?我们不可以直接修改UpperCaseInputStream的read方法,因为可能有其他人调用呢,如果是使用继承来扩展行为的话可能说需要再写一个实现类定义新的read方法,这没有必要代码也不易读。正确的做法是写一个新的装饰器来包装UpperCaseInputStream,如下:

定义一个有+1行为的装饰者:

public class UpperAndIncreaseInputStream extends FilterInputStream{

protected UpperAndIncreaseInputStream(InputStream in) {
    super(in);
}
public int read(byte b[], int off, int len) throws IOException {
    int result = super.read(b,off,len);
    for(int i =0;i<b.length;i++) {
        b[i] = (byte)(b[i]+1);
    }
    return result;
}
}

使用的时候就可以通过两层装饰实现添加+1并大写的行为:

public class InputTest {
public static void main(String[] args) {
    try {
        byte[] c  = new byte[10];
        InputStream inputStream = new UpperAndIncreaseInputStream(
                new UpperCaseInputStream(new FileInputStream("/qwer.txt")));

        inputStream.read(c,0,10);
        for (byte c1: c) {
            System.out.print((char) c1);
        }
    } catch (Exception e) {
        e.printStackTrace();
    }
}
} 

相关文章

  • 如何利用装饰者模式在不改变原有对象的基础上扩展功能

    目录 什么是装饰者模式 普通示例 装饰者模式示例 类图关系 装饰者模式使用场景 装饰者模式优点 装饰者模式缺点 什...

  • 装饰者模式

    装饰者模式 装饰者模式和适配器模式对比 装饰者模式 是一种特别的适配器模式 装饰者与被装饰者都要实现同一个接口,主...

  • java IO 的知识总结

    装饰者模式 因为java的IO是基于装饰者模式设计的,所以要了解掌握IO 必须要先清楚什么事装饰者模式(装饰者模式...

  • 设计模式-装饰者模式

    装饰者模式概念: 装饰者模式又名包装(Wrapper)模式。装饰者模式以对客户端透明的方式扩展对象的功能,是继承关...

  • java - 装饰者模式

    装饰者模式 装饰者模式:动态将责任添加到对象上。如果需要扩展功能,装饰者提供了比继承更有弹性的解决方案。装饰者模式...

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

    What: 装饰者模式又名包装(Wrapper)模式。装饰者模式动态地将责任附加到对象身上。若要扩展功能,装饰者提...

  • 装饰者(Decorator)模式

    装饰者(Decorator)模式装饰模式又名包装(Wrapper)模式。装饰模式是继承关系的一个替代方案。装饰模式...

  • 2、装饰者模式

    装饰者模式 一、基本概念 二、结构 三、案例1、装饰者模式案例2、JavaIO中使用装饰者模式 四、总结 一、基本...

  • PHP的设计模式-装饰者模式

    装饰者模式 装饰者模式 装饰者模式类似蛋糕,有草莓味、奶酪等种类,但是它们的核心都是蛋糕。 不断地将对象添加装饰的...

  • 设计模式 | 装饰者模式及典型应用

    前言 本文的主要内容: 介绍装饰者模式 示例 源码分析装饰者模式的典型应用Java I/O 中的装饰者模式spri...

网友评论

      本文标题:装饰者模式

      本文链接:https://www.haomeiwen.com/subject/jrxgzxtx.html