设计模式九:装饰器模式

文章目录

1、装饰器模式

装饰器模式(Decorator Pattern)允许向一个现有的对象添加新的功能,同时又不改变其结构。这种类型的设计模式属于结构型模式,它是作为现有的类的一个包装。

角色:

  • 抽象组件角色(Component): 定义可以动态添加任务的对象的接口

  • 具体组件角色(ConcreteComponent):定义一个要被装饰器装饰的对象,即 Component 的具体实现

  • 抽象装饰器(Decorator): 维护对象和其子类的引用

  • 具体装饰器角色(ConcreteDecorator):向对象添加新的功能或行为特征

java 复制代码
//抽象组件角色
public interface Compont {
    void operation();
}

//具体组件角色
public class ConcreteComponent implements Compont {
    @Override
    public void operation() {
        System.out.println("我是 具体构件类");
    }
}

//抽象装饰器
public interface Decorator {
    void operation();
}

//具体装饰器角色
public class ConcreteDecoratorA implements Decorator {
    private Compont compont;

    public ConcreteDecoratorA(Compont compont) {
        this.compont = compont;
    }

    @Override
    public void operation() {
        System.out.println("我是装饰者A,我先执行一些逻辑,接下来再执行构建类的方法");
        compont.operation();
    }
}

public class ConcreteDecoratorB implements Decorator {
    private Compont compont;

    public ConcreteDecoratorB(Compont compont) {
        this.compont = compont;
    }

    @Override
    public void operation() {
        System.out.println("我是装饰者B,我先执行一些逻辑,接下来再执行构建类的方法");
        compont.operation();
    }
}

2、示例

珍珠奶茶示例,购买珍珠奶茶时,往里面加不同的小料,价格不同

java 复制代码
//奶茶接口
public interface MilkyTea {
    double cost();
}
//珍珠奶茶
public class PearlMilkyTea implements MilkyTea{
    @Override
    public double cost() {
        return 13.0;
    }
}
java 复制代码
//修饰器接口
public abstract class MilkyTeaDecorator implements MilkyTea{
    private MilkyTea milkyTea;
    public MilkyTeaDecorator(MilkyTea milkyTea){
        this.milkyTea = milkyTea;
    }
    @Override
    public double cost() {
        return milkyTea.cost();
    }
}

/**
 * 椰果奶茶
 */
public class CoconutDecorator extends MilkyTeaDecorator{

    public CoconutDecorator(MilkyTea milkyTea) {
        super(milkyTea);
    }

    @Override
    public double cost() {
        System.out.println("添加椰果");
        return super.cost() + 2.0;
    }
}


/**
 * 布丁奶茶
 */
public class PuddingDecorator extends MilkyTeaDecorator{

    public PuddingDecorator(MilkyTea milkyTea) {
        super(milkyTea);
    }

    @Override
    public double cost() {
        System.out.println("添加布丁");
        return super.cost() + 1.5;
    }
}
java 复制代码
//测试类
public class Client {
    public static void main(String[] args) {
        MilkyTea milkyTea = new PearlMilkyTea();
        milkyTea = new CoconutDecorator(milkyTea);
        milkyTea = new PuddingDecorator(milkyTea);
        System.out.println(milkyTea.cost());
    }
}

创建珍珠奶茶对象,放入不同的修饰类的构造方法中达到修饰器的效果

如果需要在 添加小料 之后还要扩展甜度,则需要

1、增加 甜度 抽象修饰类,构造方法中需要有 MilkyTeaDecorator ,并实现MilkyTea

2、增加不同的具体修饰类,继承甜度抽象类

3、在测试类中

java 复制代码
MilkyTea milkyTea = new PearlMilkyTea();
milkyTea = new CoconutDecorator(milkyTea);
milkyTea = new SevenSugerDecorator(milkyTea);

由此可见,装饰器模式的优缺点"

优点

  1. 装饰类和被装饰类可以独立发展,不会相互耦合。
  2. 相比于继承,更加的轻便、灵活。
  3. 可以动态扩展一个实现类的功能,不必修改原本代码

缺点

  1. 会产生很多的装饰类,增加了系统的复杂性。
  2. 这种比继承更加灵活机动的特性,也同时意味着装饰模式比继承易于出错,排错也很困难,对于多次装饰的对象,调试时寻找错误可能需要逐级排查,较为繁琐。

3、装饰器模式与适配器模式

适配器模式和装饰器模式都可以在原类文件方法的基础上修改、增强,但侧重点不一样,
适配器模式 重在将一个接口转换为另一个接口
装饰器模式侧重对原类文件方法的增强,但接口还是同一个接口

最直观的区别可以理解为

  • 适配器模式是为了新的接口能够复用老接口的方法,改变接口名称,不改变老接口方法的内容(当然可以增强)

  • 装饰器模式是直接增强对象的方法

4、装饰器模式和代理模式

相同点: 装饰类(代理类)与被装饰类(被代理类)都需要实现相同的接口;
不同点:

  • 1、强调的重点不同,装饰器类强调通过层层装饰来扩展附属功能,而代理模式强调对代理过程的控制;
  • 2、调用的方式不同,一般装饰器模式是通过构造器层层嵌套的形式,而代理模式隐藏被代理对象的内部细节;

5、java io流的装饰器模式

java中io的基础类中使用到了装饰器模式:

其中缓冲字节流和缓冲字符流就是普通流的装饰器,具体可以参考一下两篇文章

Java面试必知必会 ------ 全面解读 Java IO(基础篇)

Java面试必知必会 ------ 全面解读 Java IO (装饰器模式篇)

相关推荐
Yu_Lijing10 分钟前
基于C++的《Head First设计模式》笔记——生成器模式
c++·笔记·设计模式
GoodStudyAndDayDayUp15 分钟前
RUO-VUE-PRO权限关联sql
java·数据库·sql
⑩-29 分钟前
RabbitMQ 架构和工作原理?RabbitMQ 延迟队列如何实现?
java·分布式·架构·rabbitmq
子非鱼@Itfuture31 分钟前
try-catch和try-with-resources区别是什么?try{}catch(){}和try(){}catch(){}有什么好处?
java·开发语言
Nyarlathotep01131 小时前
线程创建和Thread类
java
阿波罗尼亚1 小时前
JDK17 新特性
java
独自破碎E1 小时前
【面试真题拆解】Spring事务机制
java·spring·面试
我是咸鱼不闲呀1 小时前
力扣Hot100系列21(Java)——[多维动态规划]总结(不同路径,最小路径和,最长回文子串,最长公共子序列, 编辑距离)
java·leetcode·动态规划
lihao lihao1 小时前
二分查找
java·数据结构·算法
Albert Edison1 小时前
【C++11】可变参数模板
java·开发语言·c++