目录
外观(Facade)模式
本质:封装交互、简化调用
外观(门面)模式:一个子系统的外部与其内部通信必须通过一个统一的外观(Facade)对象进行
外观类将客户端与子系统的内部分隔开,使得客户端只需与外观类打交道,而不需要与子系统内部的很多类打交道,符合迪米特法则。
迪米特法则:如果两个类不必彼此直接通信,那么这两个类就不应当发生直接的相互作用。一个类需要调用另一个类的某个方法时,可以通过第三者转发这个调用。
结构组成
外观(Facade)角色:
- 客户端调用外观角色的方法,使得外观角色将客户端发来的请求委派到相应的子系统。
- 外观角色知晓相关的(一个或多个)子系统的功能和责任。
子系统(subsystem)角色:
- 每个子系统都可以被客户端直接调用,或者被外观角色调用。
- 子系统并不知道外观的存在,对于子系统而言,外观仅仅是另外一个客户端而已。
- 可以有一个或多个子系统。每个子系统都不是一个单独的类,而是一个类的集合。
适用场景
- 客户程序与抽象类的实现部分存在很大的依赖性。
- 引入Facade将这个子系统与客户以及其他的子系统分离,为一个复杂子系统提供一个简单接口。
- 在层次化结构中,可以使用Facade模式定义系统中每一层的入口点。
- 如果子系统之间相互依赖,则让他们仅通过Facade进行通信,从而简化他们之间的依赖关系。
- 希望包装或隐藏原有系统时。
- Facade可以把原有系统作为自己的私有成员。原有系统与Facade类联系在一起,使用Facade类的客户无法看到原有的系统。
- 维护一个遗留的大型系统
- 跟踪对系统的使用时,强迫所有客户通过Facade使用系统。
优缺点
优点:屏蔽了外部客户端和系统内部模块的交互
- 代码复用:Facade可以被多个客户端调用
- 节省客户端学习时间
缺点:不符合开闭原则
实例
基金与股票
Fund是外观类,拥有各个子系统的一个引用
package com.fund;
//Facade
public class Fund {
Stock s1;
NationDebt n1;
Realty r1;
public Fund() {
s1=new Stock();
n1=new NationDebt();
r1=new Realty();
}
public void invent1(){
s1.buy();
n1.buy();
r1.buy();
System.out.println();
}
public void invent2(){
s1.sell();
n1.sell();
r1.sell();
System.out.println();
}
}
package com.fund;
public class Stock {
private String name="股票A";
private double price=4.5;
public void buy(){
System.out.print("买股票A:");
System.out.println(toString());
}
public void sell(){
System.out.print("卖股票A:");
System.out.println(toString());
}
@Override
public String toString() {
return "Stock [name=" + name + ", price=" + price + "]";
}
}
package com.fund;
public class NationDebt {
private String name="国债A";
private double price=2.5;
public void buy(){
System.out.print("买国债A:");
System.out.println(toString());
}
public void sell(){
System.out.print("卖国债A:");
System.out.println(toString());
}
@Override
public String toString() {
return "NationDebt [name=" + name + ", price=" + price + "]";
}
}
package com.fund;
public class Realty {
private String name="房地产A";
private double price=3.5;
public void buy(){
System.out.print("买房地产A:");
System.out.println(toString());
}
public void sell(){
System.out.print("卖房地产A:");
System.out.println(toString());
}
@Override
public String toString() {
return "Realty [name=" + name + ", price=" + price + "]";
}
}
package com.fund;
public class Client {
public static void main(String[] args) {
Fund fund=new Fund();
fund.invent1();
fund.invent2();
}
}