【设计模式-结构型】装饰器模式

一、什么是装饰器模式

装饰器模式(Decorator Pattern)是一种结构型设计模式,它的核心思想是在不改变原有对象结构的情况下,动态地给对象增加一些功能,从而达到扩展功能的目的。举个例子,今天在家妈妈给蒸馒头。馒头蒸的过程中,妈妈去收拾衣服了。突然想起来,馒头好了,就跟你说:"帮我把馒头拿出来。"这个时候馒头特别烫,如果不烫其实用我们的手是可以拿出来的,但是这个时候光用手拿不行。所以我们想了个办法,找了一个手套,从而把馒头拿了出来。这个过程就类似于装饰器模式,手套相当于装饰器,给手(原有对象)增加了隔热的功能,使得手能够处理原本无法直接处理的烫馒头。

二、为什么使用装饰器模式

基于上面的馒头场景,我们讨论一下为什么使用装饰器模式(为什么要戴手套):

  1. 动态扩展(撤销)功能 :当需要在运行时为对象动态添加功能时,装饰器模式是一个很好的选择。例如,比如应对热馒头去拿的时候要隔热功能,不想隔热还可以直接上手。但是装饰的多了,手不热吗?手不累吗?所示要适当添加。别为了拿馒头装了一堆东西反而得不偿失。

  2. 避免子类爆炸 :如果通过继承来扩展功能,可能会导致子类数量急剧增加,使得系统变得复杂且难以维护。装饰器模式可以避免这种情况,通过组合的方式动态添加功能(继承关系的替代)。

  3. 保持原有接口不变 :装饰器模式可以在不改变原有对象接口的情况下,增加新的功能,这使得客户端代码可以透明地使用被装饰的对象,而不需要修改客户端代码。(拿馒头的手,拿的功能不变)

三、装饰器模式示例

  1. Component(抽象组件):定义了被装饰对象的接口,所有具体的组件和装饰器都实现这个接口。

    java 复制代码
    //我就是一个手,人们定义我叫手,收能拿东西
    public interface Hand {
        void pickUp(Object object) throws Exception;
    }
  2. ConcreteComponent(具体组件):实现了Component接口的具体组件,是被装饰的对象。

    java 复制代码
    //我是一个赤裸裸真是长在身上的手,人们说手可以拿东西,我也可以
    public class BareHand implements Hand {
        @Override
        public void pickUp(Object object) throws Exception {
            if (object instanceof HotBun) {
                throw new Exception("我擦,太热了!");
            }
            System.out.println("Picked up " + object.getClass().getSimpleName() + " with bare hands.");
        }
    }
  3. Decorator(装饰器抽象类):也实现了Component接口,持有一个Component对象的引用,通过组合的方式动态地为Component对象添加新的功能。

    java 复制代码
    //其实手上没准可以加点东西 
    public abstract class HandDecorator implements Hand {
        protected Hand hand;
    
        public HandDecorator(Hand hand) {
            this.hand = hand;
        }
    
        @Override
        public void pickUp(Object object) throws Exception {
            hand.pickUp(object);
        }
    }
  4. ConcreteDecorator(具体装饰器):实现了Decorator的具体装饰器,负责给Component对象添加具体的装饰功能。

    java 复制代码
    //我是手套
    public class GloveDecorator extends HandDecorator {
        public GloveDecorator(Hand hand) {
            super(hand);
        }
    
        @Override
        public void pickUp(Object object) throws Exception {
            System.out.println("戴上手套去拿 " + object.getClass().getSimpleName());
            hand.pickUp(object);
        }
    }
  5. 客户端

    java 复制代码
    //我是一个热馒头
    public class HotBun {
        // 烫馒头的具体实现
    }
    java 复制代码
    public class Main {
        public static void main(String[] args) {
            Hand bareHand = new BareHand();
            try {
                bareHand.pickUp(new HotBun());
            } catch (Exception e) {
                System.out.println(e.getMessage());
            }
    
            // 使用手套装饰手
            Hand glovedHand = new GloveDecorator(bareHand);
            try {
                glovedHand.pickUp(new HotBun());
            } catch (Exception e) {
                System.out.println(e.getMessage());
            }
        }
    }
    
    //输出
    我擦,太热了!
    戴上手套去拿热馒头
    我擦,太热了!
相关推荐
用户298698530142 小时前
Word 文档字符级格式化:Java 实现方案详解
java·后端
笨鸟飞不快2 小时前
从单个服务到集群:一次完整的性能排查复盘
java·前端
荣码2 小时前
用Streamlit给AI应用套个界面,10行代码出Web页面
java·python
SamDeepThinking2 小时前
Java微服务练习方式
java·后端·微服务
朦胧之13 小时前
AI 编程-老项目改造篇
java·前端·后端
程序猿大帅17 小时前
别再只当调包侠了:用 Spring AI 落地 Function Calling,我被大模型硬生生砸出了三个大坑
java
程序员晓琪18 小时前
约定大于配置:基于 Java 包名自动生成 API 版本路由的最佳实践
java·spring boot·后端
Flittly18 小时前
【AgentScope Java新手村系列】(11)中断与恢复
java·spring boot·spring
众少成多积小致巨19 小时前
JNI (Java Native Interface) 技术手册中文参考指南
android·java·c++
东坡白菜19 小时前
破局全栈:前端开发的Java入门实战记录—JPA(2)
java·后端