定义
- 为大型程序的调用准备一个“窗口”。
- 可以为相互关联在一起的错综复杂的类整理出高层接口(API)。
- Facade角色可以让系统对外只有一个简单的就(API)。
- Facade角色会考虑到系统内部各个类之间的责任关系和依赖关系,按照正确的顺序调用各个类。
Facade模式中登场角色
- Facade(窗口)
Facade角色是代表构成系统的许多其他角色的“简单窗口”。Facade角色向系统外部提供高层接口(API)。
- 构成系统的许多其他角色
这些角色各自完成自己的工作。它们并不知道Facade角色。Facade角色调用其他角色进行工作,但是其他角色不会调用Facade角色。
- Client(请求者)
Client角色负责调用Facade角色(Client角色并不包含在Facade模式中)。
Facade模式的类图
拓展思路的要点
- Facade角色到底做什么工作
让复杂的东西看起来简单,什么是复杂的东西呢?就是类之间的关系和它们的使用方法。使用Facade模式可以让接口(API)变少了。
- 递归地使用Facade模式
假设现在有几个持有Facade角色的类的集合。那么,我们可以通过整合这几个集合来引入新的Facade角色。即递归地使用Facade模式。
- 每个开发人员都会喜欢创建Facade角色
因为这样会使类对外提供的功能变得简单起来,其他人使用起来也很方便,减少很多沟通成本。
相关的设计模式
可以将Abastract Factory模式看作生成复杂实例的Facade模式。因为它提供了“要想生成这个实例只需要调用这个方法就OK了”的简单接口。
有时会使用Singleton模式创建Facade角色。
- Mediator模式(第16章)
在Facade模式中,Facade角色单方面地使用其他角色来提供高层接口(API)。
而在Mediator模式中,Mediator角色作为Colleague角色间的仲裁者负责调停。可以说Facade模式是单向的,而Mediator角色是双向的。
代码
- 构成系统的许多其他角色
import java.io.FileInputStream;
import java.util.Properties;
class Database {
private Database() {
}
public static Properties getProperties(String dbname) { // 根据数据库名获取Properties
String filename = dbname + ".txt";
Properties prop = new Properties();
try {
prop.load(new FileInputStream(filename));
} catch (Exception e) {
System.out.println("Warning: " + filename + " is not found.");
}
return prop;
}
}
import java.io.IOException;
import java.io.Writer;
class HtmlWriter {
private Writer writer;
public HtmlWriter(Writer writer) { // 构造函数
this.writer = writer;
}
public void title(String title) throws IOException { // 输出标题
writer.write("<html>");
writer.write("<head>");
writer.write("<title>" + title + "</title>");
writer.write("</head>");
writer.write("<body>\n");
writer.write("<h1>" + title + "</h1>\n");
}
public void paragraph(String msg) throws IOException { // 输出段落
writer.write("<p>" + msg + "</p>");
}
public void link(String href, String caption) throws IOException { // 输出超链接
paragraph("<a href=\"" + href + "\">" + caption + "</a>");
}
public void mailto(String mailaddr, String username) throws IOException { // 输出邮件地址
link("mailto:" + mailaddr, username);
}
public void close() throws IOException { // 结束输出HTML
writer.write("</body>");
writer.write("</html>\n");
writer.close();
}
}
- Facade(窗口)
import java.io.FileWriter;
import java.util.Properties;
public class PageMaker {
private PageMaker() {
}
public static void makeWelcomePage(String mailaddr, String filename) {
try {
Properties mailprop = Database
.getProperties("D:/SuychDevelop/WorkSpace/DesignPattern/15-Facade模式/src/maildata");
String username = mailprop.getProperty(mailaddr);
HtmlWriter writer = new HtmlWriter(new FileWriter(filename));
writer.title("Welcome to " + username + "'s page!");
writer.paragraph(username + "欢迎来到" + username + "的主页。");
writer.paragraph("等着你的邮件哦!");
writer.mailto(mailaddr, username);
writer.close();
System.out.println(filename + " is created for " + mailaddr + " (" + username + ")");
} catch (Exception e) {
e.printStackTrace();
}
}
}
- Client(请求者)
public class Main {
public static void main(String[] args) {
PageMaker.makeWelcomePage("suych@suych.com", "welcome.html");
}
}
注:博客中的图片来自网上。