外观模式 (Facade Pattern)

外观模式 (Facade Pattern)

外观模式是一种 结构型设计模式,通过为子系统中的一组接口提供一个统一的高层接口,简化了子系统的使用,让复杂系统更易于访问。


原理

  1. 核心思想
    • 提供一个 统一的接口 来访问子系统中的多个接口,从而简化客户端与复杂子系统的交互。
    • 客户端只需与外观接口交互,避免直接接触子系统的复杂实现细节。
  2. 适用场景
    • 需要简化复杂系统的访问。
    • 希望子系统能够被独立使用,同时也需要一个简化的入口。
  3. 参与角色
    • Facade(外观类):为客户端提供简化的高层接口。
    • Subsystems(子系统):实现系统的实际功能,可以被外观类调用,但对子系统的调用是透明的。
    • Client(客户端):通过外观类访问子系统。

优点

  1. 简化接口:降低客户端与子系统之间的耦合性。
  2. 松散耦合:客户端无需直接依赖子系统,从而更容易维护和扩展。
  3. 更好的层次划分:为复杂系统提供一个清晰的结构。

缺点

  1. 可能导致额外的封装:如果外观类的设计不够好,可能导致新的复杂性。
  2. 单一职责问题:外观类可能成为上帝类,承担过多的职责。

示例代码

场景描述

假设有一个 家庭影院系统,包含多个子系统:投影仪、音响、DVD播放器。我们希望通过一个简化的接口来启动和关闭家庭影院,而不需要逐个操作每个子系统。


1. 定义子系统类
java 复制代码
// 投影仪子系统
public class Projector {
    public void on() {
        System.out.println("Projector is ON");
    }

    public void off() {
        System.out.println("Projector is OFF");
    }
}

// 音响子系统
public class SoundSystem {
    public void on() {
        System.out.println("Sound System is ON");
    }

    public void off() {
        System.out.println("Sound System is OFF");
    }
}

// DVD播放器子系统
public class DVDPlayer {
    public void play() {
        System.out.println("DVD Player is PLAYING");
    }

    public void stop() {
        System.out.println("DVD Player has STOPPED");
    }
}

2. 创建外观类
java 复制代码
// 外观类:家庭影院控制器
public class HomeTheaterFacade {
    private Projector projector;
    private SoundSystem soundSystem;
    private DVDPlayer dvdPlayer;

    public HomeTheaterFacade(Projector projector, SoundSystem soundSystem, DVDPlayer dvdPlayer) {
        this.projector = projector;
        this.soundSystem = soundSystem;
        this.dvdPlayer = dvdPlayer;
    }

    // 启动家庭影院
    public void startMovie() {
        System.out.println("Starting Home Theater...");
        projector.on();
        soundSystem.on();
        dvdPlayer.play();
    }

    // 关闭家庭影院
    public void stopMovie() {
        System.out.println("Stopping Home Theater...");
        dvdPlayer.stop();
        soundSystem.off();
        projector.off();
    }
}

3. 客户端代码
java 复制代码
public class FacadePatternExample {
    public static void main(String[] args) {
        // 创建子系统对象
        Projector projector = new Projector();
        SoundSystem soundSystem = new SoundSystem();
        DVDPlayer dvdPlayer = new DVDPlayer();

        // 创建外观对象
        HomeTheaterFacade homeTheater = new HomeTheaterFacade(projector, soundSystem, dvdPlayer);

        // 使用外观类操作子系统
        homeTheater.startMovie(); // 启动家庭影院
        homeTheater.stopMovie();  // 关闭家庭影院
    }
}

输出结果
text 复制代码
Starting Home Theater...
Projector is ON
Sound System is ON
DVD Player is PLAYING
Stopping Home Theater...
DVD Player has STOPPED
Sound System is OFF
Projector is OFF

UML 类图

复制代码
         +---------------------+
         |   HomeTheaterFacade  |
         +---------------------+
         | + startMovie()      |
         | + stopMovie()       |
         +---------------------+
               /     |     \
              /      |      \
   +-------------+ +-------------+ +-------------+
   |  Projector  | | SoundSystem | |  DVDPlayer  |
   +-------------+ +-------------+ +-------------+
   | + on()      | | + on()      | | + play()    |
   | + off()     | | + off()     | | + stop()    |
   +-------------+ +-------------+ +-------------+

使用场景

  1. 复杂系统的统一入口:如数据库访问、网络通信等。
  2. 模块间的解耦:简化子系统对外暴露的接口,降低耦合性。
  3. 提供默认行为:如框架中预定义的工具类。

小结

  • 外观模式通过引入外观类,降低了客户端对复杂子系统的依赖,增强了系统的模块化。
  • 它适合用于封装复杂系统,同时保持子系统的灵活性。
  • 需要注意外观类的职责划分,避免其成为"上帝类"。
相关推荐
okseekw7 分钟前
Maven从入门到实战:核心概念+配置详解+避坑指南
java·后端
tryxr10 分钟前
ReentrantLock 与 synchronized 的区别
java··reentrantlock
Java爱好狂.37 分钟前
Java面试Redis核心知识点整理!
java·数据库·redis·分布式锁·java面试·后端开发·java八股文
sheji34161 小时前
【开题答辩全过程】以 基于Java的应急安全学习平台的设计与实现为例,包含答辩的问题和答案
java·开发语言·学习
程序员小假1 小时前
我们来说一下消息的可靠性投递
java·后端
席之郎小果冻1 小时前
【04】【创建型】【聊一聊,建造者模式】
java·前端·建造者模式
原来是好奇心1 小时前
深入Spring Boot源码(四):Starter机制与依赖管理深度解析
java·源码·springboot·starter
阿杆1 小时前
如何在 Spring Boot 中接入 Amazon ElastiCache
java·数据库·redis
cheems95272 小时前
锁策略的介绍
java·开发语言