JAVA设计模式8:装饰模式,动态地将责任附加到对象上,扩展对象的功能

简介: JAVA设计模式8:装饰模式,动态地将责任附加到对象上,扩展对象的功能

一、什么是装饰模式

装饰模式是一种设计模式,属于结构型模式之一。

在 Java 中,装饰模式通过动态地将责任附加到对象上,以扩展其功能,它提供了一种比继承更灵活的方式来扩展对象的功能。

在装饰模式中,有一个基础对象(被装饰对象)和一系列装饰器(装饰对象)。

装饰器和基础对象实现同一个接口,这样它们可以相互替换。

装饰器包含一个基础对象的引用,并将新的功能添加到基础对象上。

使用装饰模式,我们可以在运行时动态地扩展对象的功能,而不需要修改已有的代码 \color{red}{在运行时动态地扩展对象的功能,而不需要修改已有的代码}在运行时动态地扩展对象的功能,而不需要修改已有的代码,它将对象的行为和功能的扩展分离开来,使得代码更加灵活可扩展

下面是一个简单的Java代码示例,请同学们复制到本地执行。

// 定义口
public interface Shape {
    void draw();
}
// 实现接口的基础对象
public class Circle implements Shape {
    @Override
    public void draw() {
        System.out.println("画一个圆形");
    }
}
// 实现接口的装饰器
public abstract class ShapeDecorator implements Shape {
    protected Shape decoratedShape;
    ShapeDecorator(Shape decoratedShape) {
        this.decoratedShape = decoratedShape;
    }
    @Override
    public void draw() {
        decoratedShape.draw();
    }
}
// 具体的装饰器
public class RedShapeDecorator extends ShapeDecorator {
    public RedShapeDecorator(Shape decoratedShape) {
        super(decoratedShape);
    }
 @Override
    public void draw() {
        decoratedShape.draw();
        setRedBorder();
    }
    private void setRedBorder() {
        System.out.println("添加红色边框");
    }
}
// 使用示例
public class Main {
    public static void main(String[] args) {
        Shape circle = new Circle();
        // 使用装饰器功能扩展
        Shape redCircle = new RedShapeDecorator(new Circle());
        Shape redRectangle = new RedShapeDecorator(new Rectangle());
        circle.draw();          // 输出:画一个圆形
        redCircle.draw();       // 输出:画一个圆形,添加红色边框
        redRectangle.draw();    // 输出:画一个长方,添加红色边框
    }
}

在这个例子中,Circle是基础对象,ShapeDecorator是装饰器,RedShapeDecorator是具体的装饰器。

使用装饰器可以在运行时为基础对象添加新的功能,而不需要修改基础对象本的代码。


二、装饰模式实例

以下是一个简单的 Java 装饰模式的实例代码,请同学们复制到本地执行。

// 定义接口
interface Shape {
    void draw();
}
// 实现接口的基础对象
class Circle implements Shape {
    @Override
    public void draw() {
        System.out.println("画一个圆形");
    }
}
// 实现接口的装饰器
abstract class ShapeDecorator implements Shape {
    protected Shape decoratedShape;
    public ShapeDecorator(Shape decoratedShape) {
        this.decoratedShape = decoratedShape;
    }
    @Override
    public void draw() {
        decoratedShape.draw();
    }
}
// 具体的装饰器
class RedShapeDecorator extends ShapeDecorator {
    public RedShapeDecorator(Shape decoratedShape) {
        super(decoratedShape);
    }
    @Override
    public void draw() {
        decoratedShape.draw();
        setRedBorder();
    }
    private void setRedBorder() {
        System.out.println("添加红色边框");
    }
}
// 使用示例
public class Main {
    public static void main(String[] args) {
        Shape circle = new Circle();
        // 使用装饰器功能扩展
        Shape redCircle = new RedShapeDecorator(new Circle());
        Shape redRectangle = new RedShapeDecorator(new Rectangle());
        circle.draw(); // 输出:画一个圆形
        redCircle.draw(); // 输出:画一个圆形,添加红色边框
        redRectangle.draw(); // 输出:画一个长方形,添加红色边框
    }
}

在这个例子中,Circle是基础对象,ShapeDecorator是装饰器,RedShapeDecorator是具体的装饰器。

使用装饰器可以在运行时为基础对象添加新的功能,而不需要修改基础对象本身的代码。

通过创建具体的装饰器并传入基础对象,可以动态地扩展对象的功能。


三、装饰模式的应用场景

装饰模式在以下 4 44 种情况下使用,请同学们有个基本的了解。

  1. 动态地给一个对象添加额外的功能,而不影响其他对象。
  2. 需要在不改变原有代码的情况下,对现有对象进行扩展。
  3. 需要通过组合而非继承来实现功能的扩展。
  4. 需要对一个对象的功能进行多次扩展,而使用继承会导致类的爆炸性增长。

当然,装饰模式的应用场景有以下 4 44 类,请同学们认真学习。

  1. IO 流操作:Java 中的 InputStream、OutputStream 等类就是典型的装饰模式的应用。通过装饰器模式,我们可以在运行时动态地为输入输出流添加新的功能,如缓冲、压缩等,而不需要修改原有的 IO 类。
  2. GUI 组件:在 GUI 应用程序中,我们经常需要为组件添加额外的装饰,如边框、滚动条等。装饰模式可以让我们在不改变组件类的情况下,动态地添加这些装饰。
  3. 日志记录:通过装饰模式,我们可以为日志记录器对象动态地添加新的功能,如加密、压缩等,而不需要修改原有的日志记录器类。
  4. 数据库操作:在数据库操作中,我们可以通过装饰模式为数据库连接对象添加额外的功能,如连接池管理、事务管理等。

总之,装饰模式在需要动态地为对象添加功能、扩展对象的行为且不改变原有代码的情况下,提供了一种灵活且可复用的方案。


四、装饰模式面试题

  1. 请解释什么是装饰模式,并举一个具体的例子。
  2. 装饰模式和继承有什么区别?
  3. 装饰模式的优点和缺点是什么?
  4. 装饰模式与适配器模式有何异同?
  5. 在什么情况下应该使用装饰模式?
  6. 装饰模式如何实现动态地给对象添加新的功能?
  7. 装饰模式中的组件接口和装饰器接口有何作用?
  8. 在装饰模式中,装饰器可以有多个吗?如果可以,如何管理多个装饰器之间的顺序?
  9. 装饰模式与代理模式有何区别?
  10. 装饰模式是否违反开闭原则?为什么?
相关文章
|
3月前
|
设计模式 网络协议 数据可视化
Java 设计模式之状态模式:让对象的行为随状态优雅变化
状态模式通过封装对象的状态,使行为随状态变化而改变。以订单为例,将待支付、已支付等状态独立成类,消除冗长条件判断,提升代码可维护性与扩展性,适用于状态多、转换复杂的场景。
374 0
|
5月前
|
缓存 安全 Java
Java反射机制:动态操作类与对象
Java反射机制是运行时动态操作类与对象的强大工具,支持获取类信息、动态创建实例、调用方法、访问字段等。它在框架开发、依赖注入、动态代理等方面有广泛应用,但也存在性能开销和安全风险。本文详解反射核心API、实战案例及性能优化策略,助你掌握Java动态编程精髓。
|
5月前
|
存储 人工智能 JavaScript
Java从作用域到对象高级应用​
本内容详细讲解了JavaScript中的作用域类型(函数作用域、块作用域、全局作用域)、作用域链、垃圾回收机制、闭包、变量提升、函数参数、数组方法、内置构造函数、对象高级知识、原型链、对象赋值、深浅拷贝、递归、异常处理及this指向等内容,全面覆盖JS核心概念与编程技巧。
67 0
|
6月前
|
存储 Java
Java对象的内存布局
在HotSpot虚拟机中,Java对象的内存布局分为三部分:对象头(Header)、实例数据(Instance Data)和对齐填充(Padding)。对象头包含Mark Word、Class对象指针及数组长度;实例数据存储对象的实际字段内容;对齐填充用于确保对象大小为8字节的整数倍。
133 0
|
7月前
|
Java 数据库连接 API
Java 对象模型现代化实践 基于 Spring Boot 与 MyBatis Plus 的实现方案深度解析
本文介绍了基于Spring Boot与MyBatis-Plus的Java对象模型现代化实践方案。采用Spring Boot 3.1.2作为基础框架,结合MyBatis-Plus 3.5.3.1进行数据访问层实现,使用Lombok简化PO对象,MapStruct处理对象转换。文章详细讲解了数据库设计、PO对象实现、DAO层构建、业务逻辑封装以及DTO/VO转换等核心环节,提供了一个完整的现代化Java对象模型实现案例。通过分层设计和对象转换,实现了业务逻辑与数据访问的解耦,提高了代码的可维护性和扩展性。
286 1
|
7月前
|
前端开发 Java 数据库连接
java bo 对象详解_全面解析 java 中 PO,VO,DAO,BO,POJO 及 DTO 等几种对象类型
Java开发中常见的六大对象模型(PO、VO、DAO、BO、POJO、DTO)各有侧重,共同构建企业级应用架构。PO对应数据库表结构,VO专为前端展示设计,DAO封装数据访问逻辑,BO处理业务逻辑,POJO是简单的Java对象,DTO用于层间数据传输。它们在三层架构中协作:表现层使用VO,业务层通过BO调用DAO处理PO,DTO作为数据传输媒介。通过在线商城的用户管理模块示例,展示了各对象的具体应用。最佳实践包括保持分层清晰、使用工具类转换对象,并避免过度设计带来的类膨胀。理解这些对象模型的区别与联系。
540 1
|
8月前
|
Java
深入JavaSE:详解Java对象的比较。
总的来说,Java对象的比较就像海洋生物的比较,有外在的,有内在的,有面对所有情况的,也有针对特殊情况的。理解并掌握这些比较方式,就能更好地驾驭Java的世界,游刃有余地操作Java对象。
154 12
|
3月前
|
JSON 网络协议 安全
【Java】(10)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
213 1
|
3月前
|
JSON 网络协议 安全
【Java基础】(1)进程与线程的关系、Tread类;讲解基本线程安全、网络编程内容;JSON序列化与反序列化
几乎所有的操作系统都支持进程的概念,进程是处于运行过程中的程序,并且具有一定的独立功能,进程是系统进行资源分配和调度的一个独立单位一般而言,进程包含如下三个特征。独立性动态性并发性。
235 1
|
4月前
|
数据采集 存储 弹性计算
高并发Java爬虫的瓶颈分析与动态线程优化方案
高并发Java爬虫的瓶颈分析与动态线程优化方案

热门文章

最新文章