门面模式 与 装饰器模式(3)

简介: 门面模式 与 装饰器模式(3)

创建一个基本的煎饼(或者叫基础套餐)BaseBattercake:


public class BaseBattercake extends Battercake{
    protected String getMsg(){ return "煎饼";}
    public int getPrice(){ return 5;}
}


然后,再创建一个扩展套餐的抽象装饰器BattercakeDecotator类:



public class BattercakeDecorator extends Battercake{
    //静态代理,委派 
    private Battercake battercake;
    public BattercakeDecorator(Battercake battercake) {
        this.battercake = battercake;
    }
    @Override 
    protected String getMsg(){ return this.battercake.getMsg();}
    @Override 
    public int getPrice(){ return this.battercake.getPrice();}
}


然后,创建鸡蛋装饰器EggDecorator类:


public class EggDecorator extends BattercakeDecorator{
    public EggDecorator(Battercake battercake) {
        super(battercake);
    }
    @Override
    protected String getMsg(){ return super.getMsg() + "1个鸡蛋";}
    @Override
    public int getPrice(){ return super.getPrice() + 1;}
}


创建香肠装饰器SausageDecorator类:


public class SauageDecorator extends BattercakeDecorator{
    public SauageDecorator(Battercake battercake) {
        super(battercake);
    }
    protected String getMsg(){ return super.getMsg() + "1根香肠";}
    public int getPrice(){ return super.getPrice() + 2;}
}


编写客户端测试代码:


public class Test {
    public static void main(String[] args) {
        //路边摊买一个煎饼
        Battercake battercake = new BaseBattercake();
        //煎饼有点小,想再加一个鸡蛋
        battercake = new EggDecorator(battercake);
        //再加一个鸡蛋
        battercake = new EggDecorator(battercake);
        //很饿,再加根香肠
        battercake = new SauageDecorator(battercake);
        //跟静态代理最大区别就是职责不同 
        //静态代理不一定要满足 is-a 的关系 
        //静态代理会做功能增强,同一个职责变得不一样
        //装饰器更多考虑是扩展 
        System.out.println(battercake.getMsg() + ",总价" + battercake.getPrice());
    }
}


运行结果:


煎饼,总价:5
煎饼+1个鸡蛋,总价:6
煎饼+1个鸡蛋+1根香肠,总价:8
煎饼+1个鸡蛋+1根香肠+1根香肠,总价:9


七、装饰器模式应用与实例

1.解决易用性问题


门面模式可以用来封装系统的底层实现,隐藏系统的复杂性,提供一组更加简单易用、更高层的接口。


比如,Linux 系统调用函数就可以看作一种“门面”。它是 Linux 操作系统暴露给开发者的一组“特殊”的编程接口,它封装了底层更基础的 Linux 内核调用。


再比如,Linux 的 Shell 命令,实际上也可以看作一种门面模式的应用。它继续封装系统调用,提供更加友好、简单的命令,让我们可以直接通过执行命令来跟操作系统交互。


2.解决性能问题


我们通过将多个接口调用替换为一个门面接口调用,减少网络通信成本,提高 App 客户端的响应速度。


3.解决分布式事务问题


在用户注册的时候,我们不仅会创建用户(在数据库 User 表中),还会给用户创建一个钱包(在数据库的 Wallet 表中)。用户注册需要支持事务,也就是说,创建用户和钱包的两个操作,要么都成功,要么都失败,不能一个成功、一个失败。


最简单的解决方案是,利用数据库事务或者 Spring 框架提供的事务(如果是 Java 语言的话),在一个事务中,执行创建用户和创建钱包这两个 SQL 操作。这就要求两个 SQL 操作要在一个接口中完成,所以,我们可以借鉴门面模式的思想,再设计一个包裹这两个操作的新接口,让新接口在一个事务中执行两个 SQL 操作。


门面模式与装饰器模式详解


目录
相关文章
|
6月前
|
JSON 前端开发 应用服务中间件
跨域请求(CORS)如何解决?
CORS 全称为(Cross-Origin Resource Sharing:跨站资源共享),跨域请求是由于浏览器的同源策略(Same-Origin Policy)引起的,那么 CORS 的产生和浏览器的同源策略有关系,我们先了解什么是同源策略。
|
安全 Android开发 Kotlin
Android经典面试题之Kotlin延迟初始化的by lazy和lateinit有什么区别?
**Kotlin中的`by lazy`和`lateinit`都是延迟初始化技术。`by lazy`用于只读属性,线程安全,首次访问时初始化;`lateinit`用于可变属性,需手动初始化,非线程安全。`by lazy`支持线程安全模式选择,而`lateinit`适用于构造函数后初始化。选择依赖于属性特性和使用场景。**
377 5
Android经典面试题之Kotlin延迟初始化的by lazy和lateinit有什么区别?
|
Java
Java 匿名函数的概念和写法
Java 匿名函数的概念和写法
217 1
|
存储 Java API
java 一文讲透集合框架(10万字博文)
java API常用工具之集合框架 全面总结,10万字深度讲解。
338 0
java 一文讲透集合框架(10万字博文)
|
存储 缓存 算法
深入理解操作系统:从进程管理到内存分配
本文深入探讨操作系统的核心组件,特别关注进程管理和内存分配机制。通过分析现代操作系统中这两个关键领域的设计原理和实现技术,文章揭示了它们如何共同确保系统资源的有效利用和任务的高效执行。我们将从理论到实践,逐步解析进程状态变迁、调度算法以及内存分配策略,旨在为读者提供对操作系统内部工作原理的深刻见解。
359 0
|
监控 安全 Go
Golang深入浅出之-Channels基础:创建、发送与接收数据
【4月更文挑战第22天】Go语言的Channels是并发通信的核心,用于Goroutines间安全高效的数据交换。本文介绍了Channels的基础知识,包括创建(无缓冲和缓冲通道)、发送与接收数据,以及如何避免常见问题。创建通道使用`make(chan T)`,发送数据用`ch <- value`,接收数据则相反。无缓冲通道需匹配的发送和接收操作,否则会阻塞。缓冲通道能暂存数据,但需注意缓冲区大小以防止死锁。关闭通道后不能发送,但能接收剩余数据,多值接收可检测通道状态。掌握这些概念和技巧,能提升Go语言并发编程的效率和稳定性。
142 1
|
存储 Java 编译器
【JVM】深入了解JVM方法区
【JVM】深入了解JVM方法区
352 0
|
前端开发 JavaScript 安全
70k Star 的 Tailwind CSS 为何这么火?(上)
70k Star 的 Tailwind CSS 为何这么火?
328 0
|
设计模式 XML JSON
二十三种设计模式全面解析-工厂模式:创造对象的魔法工厂
二十三种设计模式全面解析-工厂模式:创造对象的魔法工厂
289 0
|
设计模式
二十三种设计模式全面解析-桥接模式的高级应用:构建灵活的跨平台UI框架
二十三种设计模式全面解析-桥接模式的高级应用:构建灵活的跨平台UI框架
259 0