《图解设计模式》读书笔记7-1 facade模式

1. Facade模式简介

开发程序的过程中,随着时间的推移,类会越来越多,调用关系会越来越复杂,还会涉及到调用顺序的问题。此时我们就需要一个“窗口”,把错综复杂的关联关系和调用顺序都写好,整理起来供使用者使用。这个模式最关键的思想就是:降低难度。并由此来提高开发效率,提升程序的可靠性和可维护性。

2. 示例程序

下面一段程序介绍如何生成一个网页:先从数据库里面读取用户信息,再按顺序生成网页的head、body、欢迎词等等。虽然不够“错综复杂”,但也牵涉两个类,有调用顺序。我们要利用这个模式,创建一个窗口,一行代码完成所有调用。

2.1 类图

2.2 程序

如类图所示,Database类负责从数据库(本例中是一个txt文本)里面读取用户信息,HtmlWriter则包含了一个个API,需要按顺序调用它们来生成网页。PageMaker就是窗口,它将最复杂的调用关系和顺序包装好给使用者Main,最终Main只需要一行代码即可完成工作。

public class Database {
    private Database() {    // 防止外部new出Database的实例,所以声明为private方法
    }

    public static Properties getProperties(String dbname) { // 根据数据库名获取Properties
        String filename = dbname + ".txt";
        Properties prop = new Properties();
        try {
            prop.load(new FileInputStream(filename));
        } catch (IOException e) {
            System.out.println("Warning: " + filename + " is not found.");
        }
        return prop;
    }
}


public 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>\n");
    }
    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();
    }
}


public class PageMaker {
    private PageMaker() {
    }
    public static void makeWelcomePage(String mailaddr, String filename) {
        try {
            //先从数据库中提取出来邮箱信息
            Properties mailprop = Database.getProperties("D:\\XXXXXX\\maildata");
            String username = mailprop.getProperty(mailaddr);
            //按照固定的步骤构造网页并生成文件
            HtmlWriter writer = new HtmlWriter(new FileWriter(filename));
            writer.title("Welcome to " + username + "'s page!");
            writer.paragraph("欢迎来到" + username + "的主页。");
            writer.paragraph("等着你的邮件哦!");
            writer.mailto(mailaddr, username);
            writer.close();
            System.out.println(filename + " is created for " + 
                               mailaddr + " (" + username + ")");
        } catch (IOException e) {
            e.printStackTrace();
        }
    }
}


public class Main {
    public static void main(String[] args) {
        PageMaker.makeWelcomePage("XXX@XXX.com","test.html");
    }
}

//“数据库”
XXX@XXX.com=username

3.角色和类图

  • Facade(窗口):向下调用各种复杂的类和接口,向上提供简单的API。由PageMaker扮演此角色。
  • 构成系统的其他角色:各自完成自己工作的角色,他们不知道窗口的存在,但窗口角色会调用他们。比如本例中的HtmlWriter和Database类。
  • Client(请求者):负责调用Facade角色,本例中Main扮演此角色。

4.思路拓展

  • Facade角色的好处

关键在于降低复杂度,如果一个程序员,既要关心业务逻辑,又要关心底层API的各种复杂的组合调用方式——由此带来的只会是开发效率低,出错概率高。有句名言,应该不是鲁迅说的:在计算机世界,没有加一个中间层解决不了的问题,如果有,那就递归的加。Facade就是中间层。中间层还有一个好处:降低耦合。

  • 递归使用Facade模式

即上文所说,递归的加中间层。

posted on 2019-06-10 08:21  vplus  阅读(333)  评论(0编辑  收藏  举报