设计模式 - 结构型模式_装饰器模式

简介: 装饰器的核⼼就是再**不改原有类**的基础上给类新增功能。不改变原有类,大家会想到继承、AOP切⾯,当然这些⽅式都可以实现,但是使⽤装饰器模式会是另外⼀种思路更为灵活,可以避免继承导致的⼦类过多,也可以避免AOP带来的复杂性。 很多场景⽤到装饰器模式 `new BufferedReader(new FileReader("")); `字节流、字符流、⽂件流的内容时都⻅到了这样的代码,⼀层嵌套⼀层,⼀层嵌套⼀层,字节流转字符流等等,⽽这样⽅式的使⽤就是装饰器模式的⼀种体现。

@[toc]

在这里插入图片描述


结构型模式

结构型模式主要是解决如何将对象和类组装成较大的结构, 并同时保持结构的灵活和⾼效。

结构型模式包括:适配器、桥接、组合、装饰器、外观、享元、代理,这7类

在这里插入图片描述


概述

在这里插入图片描述

装饰器的核⼼就是再不改原有类的基础上给类新增功能。

不改变原有类,大家会想到继承、AOP切⾯,当然这些⽅式都可以实现,但是使⽤装饰器模式会是另外⼀种思路更为灵活,可以避免继承导致的⼦类过多,也可以避免AOP带来的复杂性。

很多场景⽤到装饰器模式 new BufferedReader(new FileReader("")); 字节流、字符流、⽂件流的内容时都⻅到了这样的代码,⼀层嵌套⼀层,⼀层嵌套⼀层,字节流转字符流等等,⽽这样⽅式的使⽤就是装饰器模式的⼀种体现。


Case

模拟⼀个单点登录功能扩充的场景
在这里插入图片描述

以往使⽤的 SSO 是⼀个组件化通⽤的服务,不能在⾥⾯添加需要的⽤户访问验证功能。这个时候我们可以使⽤装饰器模式,扩充原有的单点登录服务。但同时也保证原有功能不受破坏,可以继续使⽤。

【工程结构】

在这里插入图片描述

  • 模拟spring中的类: HandlerInterceptor ,实现接⼝功能 SsoInterceptor 模拟的单点登录拦截服务。

【模拟Spring的HandlerInterceptor】

public interface HandlerInterceptor {

    boolean preHandle(String request, String response, Object handler);

}

实际的单点登录开发会基于 org.springframework.web.servlet.HandlerInterceptor 实现。

【模拟单点登录功能】

public class SsoInterceptor implements HandlerInterceptor{

    public boolean preHandle(String request, String response, Object handler) {
        // 模拟获取cookie
        String ticket = request.substring(1, 8);
        // 模拟校验
        return ticket.equals("success");
    }

}
  • 模拟实现⾮常简单只是截取字符串,实际使⽤需要从 HttpServletRequest request 对象中获取 cookie 信息,解析 ticket 值做校验。
  • 在返回的⾥⾯也⾮常简单,只要获取到了 success 就认为是允许登录。

Bad Impl

此场景⼤多数实现的⽅式都会采⽤继承类

继承类的实现⽅式也是⼀个⽐较通⽤的⽅式,通过继承后重写⽅法,并发将⾃⼰的逻辑覆盖进去。如果是⼀些简单的场景且不需要不断维护和扩展的,此类实现并不会有什么,也不会导致⼦类过多。

在这里插入图片描述
⼯程结构⾮常简单,只是通过 LoginSsoDecorator 继承 SsoInterceptor ,重写⽅法功能。

public class LoginSsoDecorator extends SsoInterceptor {

    private static Map<String, String> authMap = new ConcurrentHashMap<String, String>();

    static {
        authMap.put("huahua", "queryUserInfo");
        authMap.put("doudou", "queryUserInfo");
    }

    @Override
    public boolean preHandle(String request, String response, Object handler) {

        // 模拟获取cookie
        String ticket = request.substring(1, 8);
        // 模拟校验
        boolean success = ticket.equals("success");

        if (!success) return false;

        String userId = request.substring(8);
        String method = authMap.get(userId);

        // 模拟方法校验
        return "queryUserInfo".equals(method);
    }

}
  • 以上这部分通过继承重写⽅法,将个⼈可访问哪些⽅法的功能添加到⽅法中。
  • 以上看着代码还算⽐较清晰,但如果是⽐较复杂的业务流程代码,就会很混乱。

【单元测试】

    @Test
    public void test_LoginSsoDecorator() {
        LoginSsoDecorator ssoDecorator = new LoginSsoDecorator();
        String request = "1successhuahua";
        boolean success = ssoDecorator.preHandle(request, "ewcdqwt40liuiu", "t");
        System.out.println("登录校验:" + request + (success ? " 放行" : " 拦截"));
    }

模拟的相当于登录过程中的校验操作,判断⽤户是否可登录以及是否可访问⽅法。


Better Impl

接下来使⽤装饰器模式来进⾏代码优化,也算是⼀次很⼩的重构

装饰器主要解决的是直接继承下因功能的不断横向扩展导致⼦类膨胀的问题,⽽是⽤装饰器模式后就会⽐直接继承显得更加灵活同时这样也就不再需要考虑⼦类的维护。

在装饰器模式中有四个⽐较重要点抽象出来的点;

  1. 抽象构件⻆⾊(Component) - 定义抽象接⼝
  2. 具体构件⻆⾊(ConcreteComponent) - 实现抽象接⼝,可以是⼀组
  3. 装饰⻆⾊(Decorator) - 定义抽象类并继承接⼝中的⽅法,保证⼀致性
  4. 具体装饰⻆⾊(ConcreteDecorator) - 扩展装饰具体的实现逻辑

通过以上这四项来实现装饰器模式,主要核⼼内容会体现在抽象类的定义和实现上


【工程结构】

在这里插入图片描述

【装饰器模式模型结构】

在这里插入图片描述

  • 以上是⼀个装饰器实现的类图结构,重点的类是 SsoDecorator ,这个类是⼀个抽象类主要完成了对接⼝ HandlerInterceptor 继承
  • 当装饰⻆⾊继承接⼝后会提供构造函数,⼊参就是继承的接⼝实现类即可,这样就可以很⽅便的扩展出不同功能组件

【抽象类装饰⻆⾊】

public abstract class SsoDecorator implements HandlerInterceptor {

    private HandlerInterceptor handlerInterceptor;

    private SsoDecorator(){}

    public SsoDecorator(HandlerInterceptor handlerInterceptor) {
        this.handlerInterceptor = handlerInterceptor;
    }

    public boolean preHandle(String request, String response, Object handler) {
        return handlerInterceptor.preHandle(request, response, handler);
    }

}
  • 在装饰类中有两个重点的地⽅是;1)实现了处理接⼝、2)提供了构造函数、3)覆盖了⽅法preHandle
  • 以上三个点是装饰器模式的核⼼处理部分,这样可以踢掉对⼦类继承的⽅式实现逻辑功能扩展

【装饰⻆⾊逻辑实现】

public class LoginSsoDecorator extends SsoDecorator {

    private Logger logger = LoggerFactory.getLogger(LoginSsoDecorator.class);

    private static Map<String, String> authMap = new ConcurrentHashMap<String, String>();

    static {
        authMap.put("huahua", "queryUserInfo");
        authMap.put("doudou", "queryUserInfo");
    }

    public LoginSsoDecorator(HandlerInterceptor handlerInterceptor) {
        super(handlerInterceptor);
    }

    @Override
    public boolean preHandle(String request, String response, Object handler) {
        boolean success = super.preHandle(request, response, handler);
        if (!success) return false;
        String userId = request.substring(8);
        String method = authMap.get(userId);
        logger.info("模拟单点登录方法访问拦截校验:{} {}", userId, method);
        // 模拟方法校验
        return "queryUserInfo".equals(method);
    }
}
  • 在具体的装饰类实现中,继承了装饰类 SsoDecorator ,那么现在就可以扩展⽅法 preHandle
  • preHandle 的实现中可以看到,这⾥只关⼼扩展部分的功能,同时不会影响原有类的核⼼服务,也不会因为使⽤继承⽅式⽽导致的多余⼦类,增加了整体的灵活性、

【单元测试】

    @Test
    public void test_LoginSsoDecorator() {
        LoginSsoDecorator ssoDecorator = new LoginSsoDecorator(new SsoInterceptor());
        String request = "1successhuahua";
        boolean success = ssoDecorator.preHandle(request, "ewcdqwt40liuiu", "t");
        System.out.println("登录校验:" + request + (success ? " 放行" : " 拦截"));
    }
  • 结果符合预期,扩展了对⽅法拦截的校验性

还有⼀种场景也可以使⽤装饰器。例如:之前使⽤某个实现某个接⼝接收单个消息,但由于外部的升级变为发送 list 集合消息,但你⼜不希望所有的代码类都去修改这部分逻辑。那么可以使⽤装饰器模式进⾏适配 list 集合,给使⽤者依然是 for 循环后的单个消息。


小结

使⽤装饰器模式满⾜单⼀职责原则,你可以在⾃⼰的装饰类中完成功能逻辑的扩展,⽽不影响主类,同时可以按需在运⾏时添加和删除这部分逻辑。

另外装饰器模式与继承⽗类重写⽅法,在某些时候需要按需选择,并不⼀定某⼀个就是最好。

装饰器实现的重点是对抽象类继承接⼝⽅式的使⽤,同时设定被继承的接⼝可以通过构造函数传递其实现类,由此增加扩展性并重写⽅法⾥可以实现此部分⽗类实现的功能。

就像夏天穿短裤,冬天穿棉裤,⾬天穿⾬⾐⼀样,人本身没有被改变,⽽需求却被不同的装饰⽽实现。⽣活中往往⽐⽐皆是设计,当你可以融合这部分活灵活现的例⼦到代码实现中,往往会创造出更加优雅的实现⽅式。

在这里插入图片描述

相关文章
|
4月前
|
设计模式 存储 缓存
聊聊Java设计模式-装饰器模式
装饰器模式允许向一个现有的对象添加新的功能,同时不改变其结果。比如Java 中的IO框架中,`FileInputStream`(处理文件)、`ByteArrayInputStream`(处理字节数组)、`BufferedInputStream`(带缓存的处理类)等就是对`InputStream`进行的功能扩展,这就是装饰器模式的典型应用。
52 1
聊聊Java设计模式-装饰器模式
|
1天前
|
设计模式 Java
Java设计模式-装饰器模式(10)
Java设计模式-装饰器模式(10)
|
4月前
|
设计模式 Java
Java一分钟之-设计模式:装饰器模式与代理模式
【5月更文挑战第17天】本文探讨了装饰器模式和代理模式,两者都是在不改变原有对象基础上添加新功能。装饰器模式用于动态扩展对象功能,但过度使用可能导致类数量过多;代理模式用于控制对象访问,可能引入额外性能开销。文中通过 Java 代码示例展示了两种模式的实现。理解并恰当运用这些模式能提升代码的可扩展性和可维护性。
51 1
|
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)
|
4月前
|
设计模式 存储 安全
Java设计模式---结构型模式
Java设计模式---结构型模式
|
1月前
|
设计模式 XML 存储
【六】设计模式~~~结构型模式~~~适配器模式(Java)
文章详细介绍了适配器模式(Adapter Pattern),这是一种结构型设计模式,用于将一个类的接口转换成客户期望的另一个接口,使原本不兼容的接口能够一起工作,提高了类的复用性和系统的灵活性。通过对象适配器和类适配器两种实现方式,展示了适配器模式的代码应用,并讨论了其优点、缺点以及适用场景。
|
1月前
|
设计模式 缓存 Java
【十一】设计模式~~~结构型模式~~~代理模式(Java)
文章详细介绍了代理模式(Proxy Pattern),这是一种对象结构型模式,用于给对象提供一个代理以控制对它的访问。文中阐述了代理模式的动机、定义、结构、优点、缺点和适用环境,并探讨了远程代理、虚拟代理、保护代理等不同代理形式。通过一个商务信息查询系统的实例,展示了如何使用代理模式来增加身份验证和日志记录功能,同时保持客户端代码的无差别对待。此外,还讨论了代理模式在分布式技术和Spring AOP中的应用,以及动态代理的概念。
【十一】设计模式~~~结构型模式~~~代理模式(Java)