外观(Facade)模式,又叫做门面模式,是一种通过为多个复杂的子系统提供一个一致的接口,使这些子系统更加容易被访问的模式。比如说我们日常生活中医院的分诊台,就是实现统一访问接口的特性:
一、外观模式介绍
外观模式提供一个统一接口,用来访问子系统的一系列接口,从而让子系统更容易使用。这个子系统可以有多种理解方式,它既可以是一个完整的系统,也可以是更细粒度的类或者模块。主要用在接口设计方面,下面就来看看外观模式的结构:
1.1 外观模式的结构
Facade
:外观类角色,作用是为多个子系统提供一个统一接口SubSystem1、SubSystem2、SubSystem3
:子系统角色,以及内部实现的功能Client
:客户端,通过外观类对子系统集合中的功能进行访问
从上面的结构图我们可以看出,外观模式的结构比较简单,就是相当于对一组子类功能的封装和抽象。它其实就是前面提到过的设计模式原则中“迪米特原则”的典型应用:两个有交互的系统,只暴露有限且必要的接口。
下面来看看外观模式的简单实现:
1.2 外观模式的实现
根据上面的结构图,我们可以实现如下代码:
代码语言:javascript复制/**
* @description: 外观类角色
* @author: wjw
* @date: 2022/4/1
*/
public class Facade {
private SubSystem1 subSystem1 = new SubSystem1();
private SubSystem2 subSystem2 = new SubSystem2();
private SubSystem3 subSystem3 = new SubSystem3();
public void show() {
System.out.println("我是Facade外观类");
subSystem1.method1();
subSystem2.method2();
subSystem3.method3();
}
}
/**
* @description: 子系统1
* @author: wjw
* @date: 2022/4/1
*/
public class SubSystem1 {
public void method1() {
System.out.println("我是SubSystem1的method1方法");
}
}
/**
* @description: 子系统2
* @author: wjw
* @date: 2022/4/1
*/
public class SubSystem2 {
public void method2() {
System.out.println("我是SubSystem2的method2方法");
}
}
/**
* @description: 子系统3
* @author: wjw
* @date: 2022/4/1
*/
public class SubSystem3 {
public void method3() {
System.out.println("我是SubSystem3的method3方法");
}
}
/**
* @description: 客户端类
* @author: wjw
* @date: 2022/4/1
*/
public class Client {
public static void main(String[] args) {
Facade facade = new Facade();
facade.show();
}
}
测试结果如下:
代码语言:javascript复制我是Facade外观类
我是SubSystem1的method1方法
我是SubSystem2的method2方法
我是SubSystem3的method3方法
二、外观模式的应用场景
2.1 slf4j中的应用
先举个官网的例子来了解一下slf4j: 我们对Hello World
类进行日志处理
首先在配置文件配置slf4j-api:
代码语言:javascript复制<dependency>
<groupId>org.slf4j</groupId>
<artifactId>slf4j-api</artifactId>
<version>1.7.5</version>
</dependency>
然后在类中引入slf4j:
代码语言:javascript复制import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* @description: 测试slf4j
* @author: wjw
* @date: 2022/4/1
*/
public class HelloWorld {
public static void main(String[] args) {
Logger logger = LoggerFactory.getLogger(HelloWorld.class);
logger.info("Hello world");
}
}
运行main方法我们发现得到这样的结果:
代码语言:javascript复制SLF4J: Failed to load class "org.slf4j.impl.StaticLoggerBinder".
SLF4J: Defaulting to no-operation (NOP) logger implementation
SLF4J: See http://www.slf4j.org/codes.html#StaticLoggerBinder for further details.
这是因为slf4j只是一个提供日志接口的框架,不是完成日志的具体实现。所以我们需要再配置一个具体日志框架:
代码语言:javascript复制<dependency>
<groupId>ch.qos.logback</groupId>
<artifactId>logback-classic</artifactId>
<version>1.2.10</version>
</dependency>
再次运行成功:
代码语言:javascript复制09:21:38.507 [main] INFO cn.ethan.design.facade.HelloWorld - Hello world
再来看看官网中对于slf4j的介绍:
The Simple Logging Facade for Java (SLF4J) serves as a simple facade or abstraction for various logging frameworks, such as java.util.logging, logback and reload4j. SLF4J allows the end-user to plug in the desired logging framework at deployment time. Note that SLF4J-enabling your library/application implies the addition of only a single mandatory dependency, namely slf4j-api-1.7.36.jar.=
从名字The Simple Logging Facade Java 中的Facade我们可以知道,它就是外观模式的应用。slf4j没有替代任何日志框架,它仅仅只是标准日志框架的外观模式,在需要使用具体的日志框架,在配置文件加入即可:
参考资料
《重学Java设计模式》
http://c.biancheng.net/view/1369.html
https://www.cnblogs.com/xrq730/p/8619156.html