设计模式系列| 外观(门面)模式

2021-06-01 16:14:58 浏览数 (1)

大家好,我是狼王,一个爱打球的程序员

这是设计模式的第九篇,这篇让我们来认识一下外观(门面)模式


1、概述

外观模式是一种结构型设计模式, 能为程序库、 框架或其他复杂类提供一个简单的接口。

避免多种不相关的功能污染单一外观, 使其变成又一个复杂结构。客户端和其他外观都可使用附加外观。

2、适用场景

1)如果你需要一个指向复杂子系统的直接接口, 且该接口的功能有限, 则可以使用外观模式。外观将会提供指向子系统中最常用功能的快捷方式, 能够满足客户端的大部分需求。

2)如果需要将子系统组织为多层结构, 可以使用外观。你可以为每个层次创建一个外观, 然后要求各层的类必须通过这些外观进行交互。

3、实例

有以下场景:

代码语言:javascript复制
当前有学生子系统,该系统有三个接口,查询学生姓名,查询学生年龄,查询学生家庭地址。

有一个教学系统,要分别去调用这三个接口。

有一个成绩系统,要分别调用者三个接口。

有一个考试系统,也要分别调用这三个系统。

3.1 不使用外观模式时候

代码语言:javascript复制
/**
 * 学生
 */
public class Student {

    private String name = "狼王";

    private int age = 25;

    private String address = "上海";

    public Student(String name, int age, String address) {
        this.name = name;
        this.age = age;
        this.address = address;
    }

    public Student(){

    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }

    public int getAge() {
        return age;
    }

    public void setAge(int age) {
        this.age = age;
    }

    public String getAddress() {
        return address;
    }

    public void setAddress(String address) {
        this.address = address;
    }
}
代码语言:javascript复制
/**
 * 学生地址
 */
@Service
public class StudentAddressService implements IStudentAddress{

    @Override
    public String getAddress() {
        Student student = new Student();
        return student.getAddress();
    }
}
代码语言:javascript复制
/**
 * 年龄接口
 */
@Service
public class StudentAgeService implements IStudentAge{

    @Override
    public int getAge() {
        Student student = new Student();
        return student.getAge();
    }
}
代码语言:javascript复制
@Service
public class StudentNameService implements IStudentName{

    @Override
    public String getName() {
        Student student = new Student();
        return student.getName();
    }
}

三个外部服务

代码语言:javascript复制
/**
 * 教育服务
 */
@Service
public class EduService {

    @Autowired
    private StudentNameService studentNameService;

    @Autowired
    private StudentAgeService studentAgeService;

    @Autowired
    private StudentAddressService studentAddressService;

    public void getStudentName(){
        System.out.println("学生姓名是:"   studentNameService.getName());
    }

    public void getStudentAge(){
        System.out.println("学生年龄是:"   studentAgeService.getAge());
    }

    public void getStudentAddress(){
        System.out.println("学生地址是:"   studentAddressService.getAddress());
    }
}
代码语言:javascript复制
/**
 * 考试服务
 */
@Service
public class ExamService {

    @Autowired
    private StudentNameService studentNameService;

    @Autowired
    private StudentAgeService studentAgeService;

    @Autowired
    private StudentAddressService studentAddressService;

    public void getStudentName(){
        System.out.println("学生姓名是:"   studentNameService.getName());
    }

    public void getStudentAge(){
        System.out.println("学生年龄是:"   studentAgeService.getAge());
    }

    public void getStudentAddress(){
        System.out.println("学生地址是:"   studentAddressService.getAddress());
    }
}
代码语言:javascript复制
/**
 * 成绩服务
 */
@Service
public class ScoreService {

    @Autowired
    private StudentNameService studentNameService;

    @Autowired
    private StudentAgeService studentAgeService;

    @Autowired
    private StudentAddressService studentAddressService;

    public void getStudentName(){
        System.out.println("学生姓名是:"   studentNameService.getName());
    }

    public void getStudentAge(){
        System.out.println("学生年龄是:"   studentAgeService.getAge());
    }

    public void getStudentAddress(){
        System.out.println("学生地址是:"   studentAddressService.getAddress());
    }
}

3.2 使用外观模式

在学生服务这里增加一个外观service

代码语言:javascript复制
/**
 * 外观模式服务
 */
@Service
public class StudentFacedService {

    @Autowired
    private StudentNameService studentNameService;

    @Autowired
    private StudentAgeService studentAgeService;

    @Autowired
    private StudentAddressService studentAddressService;

    public String getStudentName(){
        return studentNameService.getName();
    }

    public int getStudentAge(){
        return studentAgeService.getAge();
    }

    public String getStudentAddress(){
        return studentAddressService.getAddress();
    }

}

三个调用服务只需要引入外观服务

代码语言:javascript复制
/**
 * 教育服务
 */
@Service
public class EduService {

    @Autowired
    private StudentFacedService studentFacedService;

    public void getStudentName() {
        System.out.println("学生姓名是:"   studentFacedService.getStudentName());
    }

    public void getStudentAge() {
        System.out.println("学生年龄是:"   studentFacedService.getStudentAge());
    }

    public void getStudentAddress() {
        System.out.println("学生地址是:"   studentFacedService.getStudentAddress());
    }
}
代码语言:javascript复制
/**
 * 考试服务
 */
@Service
public class ExamService {

    @Autowired
    private StudentFacedService studentFacedService;

    public void getStudentName() {
        System.out.println("学生姓名是:"   studentFacedService.getStudentName());
    }

    public void getStudentAge() {
        System.out.println("学生年龄是:"   studentFacedService.getStudentAge());
    }

    public void getStudentAddress() {
        System.out.println("学生地址是:"   studentFacedService.getStudentAddress());
    }
}
代码语言:javascript复制
/**
 * 成绩服务
 */
@Service
public class ScoreService {

    @Autowired
    private StudentFacedService studentFacedService;

    public void getStudentName() {
        System.out.println("学生姓名是:"   studentFacedService.getStudentName());
    }

    public void getStudentAge() {
        System.out.println("学生年龄是:"   studentFacedService.getStudentAge());
    }

    public void getStudentAddress() {
        System.out.println("学生地址是:"   studentFacedService.getStudentAddress());
    }
}

4、分析

以上两种方式代码结构如下所示:

从上面两张图可以看到,对于外部服务来说,极大的缩减了代码复杂度,只需要调用学生服务的一个接口。

5、总结

优点:

让客户端代码独立独立于复杂的子系统,且减少对于子系统的依赖。

缺点:

过于庞大的外观,会使得该外观称成为上帝对象,造成所有类的耦合,可通过它操作所有的类功能。

好了。今天就说到这了,我还会不断分享自己的所学所想,希望我们一起走在成功的道路上!

0 人点赞