Facade 门面模式 外观模式

时间:2021-06-01 05:45:23

简介
作用:
(1)封装一组交互类,一致地对外提供接口
(2)封装子系统,简化子系统调用
JDK中体现:java.util.logging包
java.lang.Class
javax.faces.webapp.FacesServlet

医院的例子
如果把医院作为一个子系统,按照部门职能,这个系统可以划分为挂号、门诊、划价、化验、收费、取药等。看病的病人要与这些部门打交道,就如同一个子系统的客户端与一个子系统的各个类打交道一样,不是一件容易的事情。
首先病人必须先挂号,然后门诊。如果医生要求化验,病人必须首先划价,然后缴款,才能到化验部门做化验。化验后,再回到门诊室。
解决这种不便的方法便是引进门面模式。可以设置一个接待员的位置,由接待员负责代为挂号、划价、缴费、取药等。这个接待员就是门面模式的体现,病人只接触接待员,由接待员负责与医院的各个部门打交道。
什么是门面模式
门面模式要求一个子系统的外部与其内部的通信必须通过一个统一的门面(Facade)对象进行。门面模式提供一个高层次的接口,使得子系统更易于使用。
就如同医院的接待员一样,门面模式的门面类将客户端与子系统的内部复杂性分隔开,使得客户端只需要与门面对象打交道,而不需要与子系统内部的很多对象打交道。

在什么情况下使用门面模式
为一个复杂子系统提供一个简单接口
提高子系统的独立性
在层次化结构中,可以使用Facade模式定义系统中每一层的入口。

演示
对于一个复杂的系统,可能需要多个业务才能完成,如把大象关进冰箱里面
public interface IServiceA {
    void methodA();
}
public interface IServiceB {
    void methodB();
}
public class ServiceAImpl implements IServiceA {
    @Override
    public void methodA() {
        System.out.println("打开冰箱门");
    }
}
public class ServiceBImpl implements IServiceB {
    @Override
    public void methodB() {
        System.out.println("关闭冰箱门");
    }
}

这种情况下,使用者必须知道系统各个业务的具体功能及先后顺序才能使用
public class Test {
    public static void main(String[] args) {
        IServiceA sa = new ServiceAImpl();
        IServiceB sb = new ServiceBImpl();
        sa.methodA();
        sb.methodB();
    }
}


下面我们定义一个类,对子系统进行封装,以简化子系统调用
/**
 * 给一组组件,接口,抽象,或者子系统提供一个简单的访问接口
 */
public class Facade {
    private IServiceA sa;
    private IServiceB sb;
    private IServiceC sc;
    public Facade() {
        sa = new ServiceAImpl();
        sb = new ServiceBImpl();
        sc = new ServiceCImpl();
    }
    /**
     * 把大象装到冰箱里面
     */
    public void storeElephantidae() {
        sa.methodA();
        sb.methodB();
    }
    /**
     * 把大象装到冰箱里面,并冷冻起来
     */
    public void storeElephantidae2() {
        sa.methodA();
        sb.methodB();
        sc.methodC();
    }
}

这是一个很好的封装方法,如果一个子系统比较复杂的话,就可以封装出一个或多个门面出来,可以使项目的结构更简单,而且扩展性非常好。
public class Test {
    public static void main(String[] args) {
        IServiceA sa = new ServiceAImpl();
        IServiceB sb = new ServiceBImpl();
        sa.methodA();
        sb.methodB();
        System.out.println("================");
        Facade facade = new Facade();
        facade.storeElephantidae();
        System.out.println("================");
        facade.storeElephantidae2();
    }
}

附件列表