Java设计模式之装饰器模式

简介: Java设计模式之装饰器模式


文章目录

概念

装饰器模式指的是在不必改变原类文件和不使用继承的情况下,动态地扩展一个对象的功能。它是通过创建一个包装对象,也就是装饰来包裹真实的对象。

一般情况下想要扩展一个类的功能需要使用继承来实现,比如说HashSet有保证元素不重复的特点,但是它是无序存储,即存的顺序和取出的顺序不一致,为了让他在不修改原来对象的情况下实现有序存储,Java类库里面实现了一个继承HashSet类的子类LinkedHashSet

但是如果想要拓展的这个类有许多子类,那么有多少个子类就要创建多少个新的类,这会使程序变得十分复杂。

使用装饰器模式就可以解决这个问题。使用装饰器模式扩展一个类的功能我们只需要用装饰类来包裹被扩展的类。

代码演示

下面我们用代码模拟装饰器模式。

首先我们先创建一个笔记本电脑类。

//定义了一个笔记本电脑的抽象类
public abstract class Laptop {
    protected String name;
    protected int price;
    public abstract  String getName();
    public abstract  int getPrice();
}

我们创建一个联想电脑类和华硕电脑类,这两个类就是要被装饰的类。

public class Lenovo extends Laptop{
    public Lenovo(){
        this.name = "联想电脑";
        this.price = 5000;
    }
    @Override
    public String getName() {
        return name;
    }
    @Override
    public int getPrice() {
        return price;
    }
}
public class ASUS extends Laptop {
    public ASUS() {
        this.name = "华硕";
        this.price = 5500;
    }
    @Override
    public String getName() {
        return name;
    }
    @Override
    public int getPrice() {
        return price;
    }
}

创建一个装饰类基类,主要这里有一个要点,装饰类和被装饰类一定要有相同的父类或者间接父类,后面会解释原因。

//电脑配件的基类
public abstract class AddParts extends Laptop{
  //这里必须创建被装饰类的实例变量,只有这样才能做到装饰类包裹(装饰)被装饰类。
    Laptop laptop;
    public abstract String getName();
    public abstract int getPrice();
}

在这里我们创建两个装饰类:添加显示器和添加键盘。

public class AddMonitor extends AddParts {
    public AddMonitor(Laptop laptop) {
        this.laptop = laptop;
    }
    @Override
    public String getName() {
        return laptop.getName() + "   "+"显示器";
    }
    @Override
    public int getPrice() {
        return laptop.getPrice()+1000;
    }
}
public class AddKeyBoard extends AddParts {
    public AddKeyBoard(Laptop laptop) {
        this.laptop = laptop;
    }
    @Override
    public String getName() {
        return laptop.getName() +"  "+"键盘";
    }
    @Override
    public int getPrice() {
        return laptop.getPrice() + 200;
    }
}

接下来我们进行测试

public class Test {
    public static void main(String[] args) {
        //一台普通联想电脑
        Lenovo lenovo = new Lenovo();
        System.out.println(lenovo.getName() + "  "+lenovo.getPrice());
        //一台普通华硕电脑
        ASUS asus = new ASUS();
        System.out.println(asus.getName() + "  " +asus.getPrice());
        //一台装了键盘的联想电脑
        AddKeyBoard lenovo2 = new AddKeyBoard(new Lenovo());
        System.out.println(lenovo2.getName()+"  "+lenovo2.getPrice());
        //一台装了显示器和键盘的华硕电脑
        AddKeyBoard asus2 = new AddKeyBoard(new AddMonitor(new ASUS()));
        System.out.println(asus2.getName() + "  "+ asus2.getPrice());
    }
} 

运行结果

通过使用不同的具体装饰类以及这些装饰类的排列组合,设计师可以创造出很多不同行为的组合。如果不使用装饰器模式,又不想改变原有的类,使用继承来拓展类的功能,那么一种组合方式就对应一个新的类。

前面我们还提到一个要点:装饰类和被装饰类一定要有相同的父类或者间接父类。

因为只有这样才能让被装饰类被多个装饰类装饰,即让电脑同时装上显示器和键盘。

模式特点

这里我们总结一下装饰器模式的特点:

  1. 装饰类和被装饰类一定要有相同的父类或者间接父类。
  2. 装饰类包含一个真实对象的引用(reference),如下图所示。
  3. 在不改变原先类的情况下,通过使用不同的具体装饰类以及这些装饰类的排列组合,设计师可以创造出很多不同行为的组合。

设计原则

  1. 多组合,少继承。
    利用继承设计子类的行为,是在编译时静态决定的,而且所有的子类都会继承到相同的行为。然而,如果能够利用组合的做法扩展对象的行为,就可以在运行时动态地进行扩展。
  2. 类应设计的对扩展开放,对修改关闭。

设计原则来自于百度百科

装饰器模式的适用性

  1. 需要动态的给一个对象添加功能,这些功能可以再动态的撤销。
  2. 当不能采用生成子类的方法进行扩充时。一种情况是,可能有大量独立的扩展,为支持每一种组合将产生大量的子类,使得子类数目呈爆炸性增长。另一种情况可能是因为类定义被隐藏,或类定义不能用于生成子类。

Java-IO中的装饰器模式

在上面的例子中,我们是用AddParts作为装饰类的基类来增强Laptop的功能,在Java中,FilterInputStream就是用来增强InputStream的。

下面我们以FilterInputStreamBufferedInputStream为例讲解。

FilterInputStream作为装饰器加强了BufferedInputStream,下面是他们的部分源代码,加上了我的注释。

下面是被装饰类,相当于例子中的 LenovoASUS

public class FileInputStream extends InputStream{
}

下面是装饰器的基类,相当于AddParts,它与FileInputStream有着共同父类,类中有该父类的引用。

public class FilterInputStream extends InputStream {
  //装饰类包含一个真实对象的引用(reference)
  protected volatile InputStream in;
}

下面是装饰类,继承了装饰器基类,相当于例子中的加键盘和加显示器。

public class BufferedInputStream extends FilterInputStream {
  public BufferedInputStream(InputStream in) {
        this(in, DEFAULT_BUFFER_SIZE);
    }
    public BufferedInputStream(InputStream in, int size) {
        super(in);
        if (size <= 0) {
            throw new IllegalArgumentException("Buffer size <= 0");
        }
        buf = new byte[size];
    }

下面是对BufferedInputStream的使用。

public class Test {
    public static void main(String[] args) throws FileNotFoundException {
        BufferedInputStream bfs = new BufferedInputStream(new FileInputStream("text.txt"));
    }
}

学习心得

这是我第一次接触Java中的设计模式,刚开始看的时候感觉很难理解,当自己动手去模拟实现这个过程的时候,才慢慢体会到这种设计模式的巧妙,所以编程语言的学习还是多多动手,下面是我学习过程中参考的资料。

JavaGuide------装饰器模式

百度百科

设计模式系列7-装饰器模式

JAVA设计模式初探之装饰者模式


相关文章
|
1天前
|
设计模式 存储 Java
【十】设计模式~~~结构型模式~~~享元模式(Java)
文章详细介绍了享元模式(Flyweight Pattern),这是一种对象结构型模式,通过共享技术实现大量细粒度对象的重用,区分内部状态和外部状态来减少内存中对象的数量,提高系统性能。通过围棋棋子的设计案例,展示了享元模式的动机、定义、结构、优点、缺点以及适用场景,并探讨了单纯享元模式和复合享元模式以及与其他模式的联用。
【十】设计模式~~~结构型模式~~~享元模式(Java)
|
1天前
|
设计模式 存储 Java
【九】设计模式~~~结构型模式~~~外观模式(Java)
文章详细介绍了外观模式(Facade Pattern),这是一种对象结构型模式,通过引入一个外观类来简化客户端与多个子系统之间的交互,降低系统的耦合度,并提供一个统一的高层接口来使用子系统。通过文件加密模块的实例,展示了外观模式的动机、定义、结构、优点、缺点以及适用场景,并讨论了如何通过引入抽象外观类来提高系统的可扩展性。
【九】设计模式~~~结构型模式~~~外观模式(Java)
|
1天前
|
设计模式 Java
【八】设计模式~~~结构型模式~~~装饰模式(Java)
文章详细介绍了装饰模式(Decorator Pattern),这是一种对象结构型模式,用于在不使用继承的情况下动态地给对象添加额外的职责。装饰模式通过关联机制,使用装饰器类来包装原有对象,并在运行时通过组合的方式扩展对象的行为。文章通过图形界面构件库的设计案例,展示了装饰模式的动机、定义、结构、优点、缺点以及适用场景,并提供了Java代码实现和应用示例。装饰模式提高了系统的灵活性和可扩展性,适用于需要动态、透明地扩展对象功能的情况。
【八】设计模式~~~结构型模式~~~装饰模式(Java)
|
1天前
|
设计模式 XML 存储
【七】设计模式~~~结构型模式~~~桥接模式(Java)
文章详细介绍了桥接模式(Bridge Pattern),这是一种对象结构型模式,用于将抽象部分与实现部分分离,使它们可以独立地变化。通过实际的软件开发案例,如跨平台视频播放器的设计,文章阐述了桥接模式的动机、定义、结构、优点、缺点以及适用场景,并提供了完整的代码实现和测试结果。桥接模式适用于存在两个独立变化维度的系统,可以提高系统的可扩展性和灵活性。
【七】设计模式~~~结构型模式~~~桥接模式(Java)
|
1天前
|
设计模式 存储 负载均衡
【五】设计模式~~~创建型模式~~~单例模式(Java)
文章详细介绍了单例模式(Singleton Pattern),这是一种确保一个类只有一个实例,并提供全局访问点的设计模式。文中通过Windows任务管理器的例子阐述了单例模式的动机,解释了如何通过私有构造函数、静态私有成员变量和公有静态方法实现单例模式。接着,通过负载均衡器的案例展示了单例模式的应用,并讨论了单例模式的优点、缺点以及适用场景。最后,文章还探讨了饿汉式和懒汉式单例的实现方式及其比较。
【五】设计模式~~~创建型模式~~~单例模式(Java)
|
1天前
|
设计模式 算法 安全
Java编程中的设计模式:提升代码的可维护性和扩展性
【8月更文挑战第19天】在软件开发的世界里,设计模式是解决常见问题的一种优雅方式。本文将深入探讨Java编程语言中常用的几种设计模式,并解释如何通过这些模式来提高代码的可维护性和扩展性。文章不涉及具体的代码实现,而是侧重于理论和实践相结合的方式,为读者提供一种思考和改善现有项目的新视角。
|
1天前
|
设计模式 XML 存储
【六】设计模式~~~结构型模式~~~适配器模式(Java)
文章详细介绍了适配器模式(Adapter Pattern),这是一种结构型设计模式,用于将一个类的接口转换成客户期望的另一个接口,使原本不兼容的接口能够一起工作,提高了类的复用性和系统的灵活性。通过对象适配器和类适配器两种实现方式,展示了适配器模式的代码应用,并讨论了其优点、缺点以及适用场景。
|
1天前
|
设计模式 Java
常用设计模式介绍~~~ Java实现 【概念+案例+代码】
文章提供了一份常用设计模式的全面介绍,包括创建型模式、结构型模式和行为型模式。每种设计模式都有详细的概念讲解、案例说明、代码实例以及运行截图。作者通过这些模式的介绍,旨在帮助读者更好地理解源码、编写更优雅的代码,并进行系统重构。同时,文章还提供了GitHub上的源码地址,方便读者直接访问和学习。
常用设计模式介绍~~~ Java实现 【概念+案例+代码】
|
1天前
|
设计模式 算法 Java
【十六】设计模式~~~行为型模式~~~策略模式(Java)
文章详细介绍了策略模式(Strategy Pattern),这是一种对象行为型模式,用于定义一系列算法,将每个算法封装起来,并使它们可以相互替换。策略模式让算法独立于使用它的客户而变化,提高了系统的灵活性和可扩展性。通过电影院售票系统中不同类型用户的打折策略案例,展示了策略模式的动机、定义、结构、优点、缺点以及适用场景,并提供了Java代码实现和测试结果。
【十六】设计模式~~~行为型模式~~~策略模式(Java)
|
1天前
|
设计模式 网络协议 Java
【十五】设计模式~~~行为型模式~~~状态模式(Java)
文章详细介绍了状态模式(State Pattern),这是一种对象行为型模式,用于处理对象在其内部状态改变时的行为变化。文中通过案例分析,如银行账户状态管理和屏幕放大镜工具,展示了状态模式的应用场景和设计方法。文章阐述了状态模式的动机、定义、结构、优点、缺点以及适用情况,并提供了Java代码实现和测试结果。状态模式通过将对象的状态和行为封装在独立的状态类中,提高了系统的可扩展性和可维护性。
【十五】设计模式~~~行为型模式~~~状态模式(Java)