外观模式及运用场景

外观模式(Facade Pattern)是一种结构性设计模式,它为复杂子系统提供一个统一的接口,从而简化与这些子系统的交互。通过外观模式,客户端可以更轻松地使用复杂系统,而不必了解其内部实现。接下来,我们将通过示例代码详细讲解外观模式,并探讨它的应用场景。

外观模式示例代码

在这个示例中,我们将创建一个简单的音响系统。这个系统包括多个子系统,如 DVD 播放器、CD 播放器和音响设备。我们会为这个复杂系统提供一个外观类,以简化用户的操作。

示例代码

java 复制代码
// DVD 播放器子系统
class DVDPlayer {
    public void on() {
        System.out.println("DVD 播放器开机");
    }

    public void play() {
        System.out.println("DVD 播放中...");
    }

    public void off() {
        System.out.println("DVD 播放器关机");
    }
}

// CD 播放器子系统
class CDPlayer {
    public void on() {
        System.out.println("CD 播放器开机");
    }

    public void play() {
        System.out.println("CD 播放中...");
    }

    public void off() {
        System.out.println("CD 播放器关机");
    }
}

// 音响设备子系统
class SoundSystem {
    public void on() {
        System.out.println("音响设备开机");
    }

    public void setVolume(int level) {
        System.out.println("音量设置为 " + level);
    }

    public void off() {
        System.out.println("音响设备关机");
    }
}

// 外观类
class HomeTheaterFacade {
    private DVDPlayer dvdPlayer;
    private CDPlayer cdPlayer;
    private SoundSystem soundSystem;

    public HomeTheaterFacade() {
        dvdPlayer = new DVDPlayer();
        cdPlayer = new CDPlayer();
        soundSystem = new SoundSystem();
    }

    public void watchMovie() {
        soundSystem.on();
        soundSystem.setVolume(10);
        dvdPlayer.on();
        dvdPlayer.play();
    }

    public void listenToCD() {
        soundSystem.on();
        soundSystem.setVolume(5);
        cdPlayer.on();
        cdPlayer.play();
    }

    public void turnOff() {
        dvdPlayer.off();
        cdPlayer.off();
        soundSystem.off();
    }
}

// 客户端代码
public class Main {
    public static void main(String[] args) {
        HomeTheaterFacade homeTheater = new HomeTheaterFacade();
        homeTheater.watchMovie();
        homeTheater.listenToCD();
        homeTheater.turnOff();
    }
}

外观模式的运用场景

外观模式可以在多个实际应用场景中发挥重要作用。以下是几个常见的运用场景及其相应的代码示例。

1. 简化复杂系统

在电商系统中,用户需要完成商品购买,这可能涉及订单、商品、支付和会员等多个子系统。我们可以通过外观类来简化用户的操作。

示例代码
java 复制代码
// 商品子系统
class ProductService {
    public void listProducts() {
        System.out.println("列出所有商品");
    }
}

// 订单子系统
class OrderService {
    public void createOrder(String product) {
        System.out.println("创建订单,商品: " + product);
    }
}

// 支付子系统
class PaymentService {
    public void processPayment(String orderId) {
        System.out.println("处理订单支付,订单ID: " + orderId);
    }
}

// 电商外观类
class ECommerceFacade {
    private ProductService productService;
    private OrderService orderService;
    private PaymentService paymentService;

    public ECommerceFacade() {
        productService = new ProductService();
        orderService = new OrderService();
        paymentService = new PaymentService();
    }

    public void purchaseProduct(String product) {
        productService.listProducts();
        orderService.createOrder(product);
        paymentService.processPayment("12345");
    }
}

// 客户端代码
public class Main {
    public static void main(String[] args) {
        ECommerceFacade facade = new ECommerceFacade();
        facade.purchaseProduct("手机");
    }
}

2. 减少客户端处理的系统数量

在 Web 应用中,客户端可能需要处理多个系统的调用,如数据库、业务对象等。通过创建一个 DAO 层作为门面,可以简化这些操作。

示例代码
java 复制代码
// 模拟数据库操作
class Database {
    public void connect() {
        System.out.println("连接数据库");
    }

    public void disconnect() {
        System.out.println("关闭数据库连接");
    }

    public String fetchData() {
        return "数据记录";
    }
}

// 业务对象
class BusinessModel {
    private String data;

    public BusinessModel(String data) {
        this.data = data;
    }

    public void display() {
        System.out.println("业务数据: " + data);
    }
}

// DAO层
class DataAccessFacade {
    private Database database;

    public DataAccessFacade() {
        database = new Database();
    }

    public BusinessModel getBusinessModel() {
        database.connect();
        String data = database.fetchData();
        database.disconnect();
        return new BusinessModel(data);
    }
}

// 客户端代码
public class Main {
    public static void main(String[] args) {
        DataAccessFacade dataAccess = new DataAccessFacade();
        BusinessModel model = dataAccess.getBusinessModel();
        model.display();
    }
}

3. 让一个系统为多个系统工作

线程池是外观模式的另一个典型应用,它为系统提供统一的线程管理接口,使得线程的创建和使用变得简单。

示例代码
java 复制代码
import java.util.concurrent.ExecutorService;
import java.util.concurrent.Executors;

// 线程池外观
class ThreadPoolFacade {
    private ExecutorService executorService;

    public ThreadPoolFacade(int poolSize) {
        executorService = Executors.newFixedThreadPool(poolSize);
    }

    public void executeTask(Runnable task) {
        executorService.execute(task);
    }

    public void shutdown() {
        executorService.shutdown();
    }
}

// 客户端代码
public class Main {
    public static void main(String[] args) {
        ThreadPoolFacade threadPool = new ThreadPoolFacade(5);
        
        for (int i = 0; i < 10; i++) {
            final int taskId = i;
            threadPool.executeTask(() -> {
                System.out.println("执行任务 " + taskId);
            });
        }
        
        threadPool.shutdown();
    }
}

总结

外观模式通过提供一个统一的接口,简化了复杂系统的使用,减少了客户端与多个子系统之间的直接交互。在电商系统、数据库访问和线程管理等场景中,外观模式都能有效提高系统的可维护性和可读性。合理运用外观模式,可以显著提升开发效率和用户体验。

相关推荐
V+zmm101345 分钟前
基于微信小程序的乡村政务服务系统springboot+论文源码调试讲解
java·微信小程序·小程序·毕业设计·ssm
Oneforlove_twoforjob30 分钟前
【Java基础面试题025】什么是Java的Integer缓存池?
java·开发语言·缓存
xmh-sxh-131432 分钟前
常用的缓存技术都有哪些
java
AiFlutter1 小时前
Flutter-底部分享弹窗(showModalBottomSheet)
java·前端·flutter
J不A秃V头A2 小时前
IntelliJ IDEA中设置激活的profile
java·intellij-idea
DARLING Zero two♡2 小时前
【优选算法】Pointer-Slice:双指针的算法切片(下)
java·数据结构·c++·算法·leetcode
小池先生2 小时前
springboot启动不了 因一个spring-boot-starter-web底下的tomcat-embed-core依赖丢失
java·spring boot·后端
CodeClimb2 小时前
【华为OD-E卷-木板 100分(python、java、c++、js、c)】
java·javascript·c++·python·华为od
程序员厉飞雨2 小时前
Android R8 耗时优化
android·java·前端
odng2 小时前
IDEA自己常用的几个快捷方式(自己的习惯)
java·ide·intellij-idea