转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/8744002

今天你还是像往常一样来上班,一如既往地开始了你的编程工作。

项目经理告诉你,今天想在服务器端增加一个新功能,希望写一个方法,能对Book对象进行处理,将Book对象的所有字段以XML格式进行包装,这样以后可以方便与客户端进行交互。并且在包装开始前和结束后要打印日志,这样方便调试和问题定位。

没问题!你觉得这个功能简直是小菜一碟,非常自信地开始写起代码。

Book对象代码如下:

  1. public class Book {
  2. private String bookName;
  3. private int pages;
  4. private double price;
  5. private String author;
  6. private String isbn;
  7. public String getBookName() {
  8. return bookName;
  9. }
  10. public void setBookName(String bookName) {
  11. this.bookName = bookName;
  12. }
  13. public int getPages() {
  14. return pages;
  15. }
  16. public void setPages(int pages) {
  17. this.pages = pages;
  18. }
  19. public double getPrice() {
  20. return price;
  21. }
  22. public void setPrice(double price) {
  23. this.price = price;
  24. }
  25. public String getAuthor() {
  26. return author;
  27. }
  28. public void setAuthor(String author) {
  29. this.author = author;
  30. }
  31. public String getIsbn() {
  32. return isbn;
  33. }
  34. public void setIsbn(String isbn) {
  35. this.isbn = isbn;
  36. }
  37. }

然后写一个类专门用于将Book对象包装成XML格式:

  1. public class Formatter {
  2. public String formatBook(Book book) {
  3. System.out.println("format begins");
  4. String result = "";
  5. result += "<book_name>" + book.getBookName() + "</book_name>\n";
  6. result += "<pages>" + book.getPages() + "</pages>\n";
  7. result += "<price>" + book.getPrice() + "</price>\n";
  8. result += "<author>" + book.getAuthor() + "</author>\n";
  9. result += "<isbn>" + book.getIsbn() + "</isbn>\n";
  10. System.out.println("format finished");
  11. return result;
  12. }
  13. }

调用代码如下:

  1. public class Test {
  2. public static void main(String[] args) throws Exception {
  3. Book book = new Book();
  4. book.setBookName("Thinking in Java");
  5. book.setPages(880);
  6. book.setPrice(68);
  7. book.setAuthor("Bruce Eckel");
  8. book.setIsbn("9787111213826");
  9. Formatter formatter = new Formatter();
  10. String result = formatter.formatBook(book);
  11. System.out.println(result);
  12. }
  13. }

你写好了之后,迫不及待地开始运行,运行结果也完全符合你的期望。

项目经理看完后,对你非常满意,小伙效率很高的嘛!你也非常的得意。

不过两天之后,项目经理又找到了你,他说之前没有考虑到需要交互的客户端还包括手机设备,而手机设备都比较吃流量,用XML格式来传输太耗流量了,想最好能改成使用JSON格式传输。但是之前的XML格式也要保留,最好可以由客户端指定使用哪种格式。

你有些不开心,心里低估着,为什么一开始不考虑周全呢,现在又要改遗留代码。但对方毕竟是领导,你还是要服从命令的,于是你开始修改Formatter类:

  1. public class Formatter {
  2. public static final int XML = 0;
  3. public static final int JSON = 1;
  4. public String formatBook(Book book, int format) {
  5. System.out.println("format begins");
  6. String result = "";
  7. if (format == XML) {
  8. result += "<book_name>" + book.getBookName() + "</book_name>\n";
  9. result += "<pages>" + book.getPages() + "</pages>\n";
  10. result += "<price>" + book.getPrice() + "</price>\n";
  11. result += "<author>" + book.getAuthor() + "</author>\n";
  12. result += "<isbn>" + book.getIsbn() + "</isbn>\n";
  13. } else if (format == JSON) {
  14. result += "{\n";
  15. result += "\"book_name\" : \"" + book.getBookName() + "\",\n";
  16. result += "\"pages\" : \"" + book.getPages() + "\",\n";
  17. result += "\"price\" : \"" + book.getPrice() + "\",\n";
  18. result += "\"author\" : \"" + book.getAuthor() + "\",\n";
  19. result += "\"isbn\" : \"" + book.getIsbn() + "\",\n";
  20. result += "}";
  21. }
  22. System.out.println("format finished");
  23. return result;
  24. }
  25. }

调用代码如下:

  1. public class Test {
  2. public static void main(String[] args) throws Exception {
  3. Book book = new Book();
  4. book.setBookName("Thinking in Java");
  5. book.setPages(880);
  6. book.setPrice(68);
  7. book.setAuthor("Bruce Eckel");
  8. book.setIsbn("9787111213826");
  9. Formatter formatter = new Formatter();
  10. String result = formatter.formatBook(book, Formatter.XML);
  11. System.out.println(result);
  12. result = formatter.formatBook(book, Formatter.JSON);
  13. System.out.println(result);
  14. }
  15. }

再次运行程序,得到了以下结果。

项目经理看到运行结果后开心地说:“太好了,这正是我想要的!”

可是你这次却没有那么开心,你觉得代码已经有些混乱了,XML格式的逻辑和JSON格式的逻辑混淆在一起,非常不利于阅读,而且如果以后还需要扩展功能也会非常困难。好在传输格式一般也就XML和JSON了,应该不会再有什么扩展了,你这样安慰自己道。

但幻想总会被现实打破,“我最近听说有个YAML格式挺好玩的.......” 项目经理说道。这个时候你已经有想打人的冲动了!!!

很多时候就是这样,在公司里写的代码乱七八糟,质量极差,很大一部分原因就是因为需求变来变去。我们不断在原有代码基础上补充各种后续加入的情况,在一行行新增的if语句下面,我们的代码变得不堪入目。当然,我们作为程序员,对于需求这种东西没有太多的话语权,在这方面我们无能为力。但是我们可以尽量地把程序的架构设计好,让我们写出的代码更具有扩展性,这样就可以应对各种需求变更了。

 

下面你将要使用23种设计模式中的模板方法来改进以上程序。

首先将Formatter中的代码进行修改,如下所示:

  1. public abstract class Formatter {
  2. public String formatBook(Book book, int format) {
  3. beforeFormat();
  4. String result = formating(book);
  5. afterFormat();
  6. return result;
  7. }
  8. protected void beforeFormat() {
  9. System.out.println("format begins");
  10. }
  11. protected abstract String formating(Book book);
  12. protected void afterFormat() {
  13. System.out.println("format finished");
  14. }
  15. }

你会发现format_book方法只有四步,第一步调用before_format,去打印格式转换前的日志。第二步调用formating,这个方法是个抽象方法,用于处理具体的转换逻辑,因此每一个继承自Formatter的子类都需要重写此方法,来实现各自的转换逻辑。第三步调用after_format,去打印格式转换后的日志。第四步返回result。由于类中存在了抽象方法,我们也就需要把Formatter声明成抽象类。

然后要定义专门的子类来处理每种传输格式的具体逻辑,这样不同传输格式的逻辑可以从一个方法里分离开,明显便于阅读和理解。

定义类XMLFormatter继承自Formatter,里面加入处理XML格式的具体逻辑:

  1. public class XMLFormatter extends Formatter {
  2. @Override
  3. protected String formating(Book book) {
  4. String result = "";
  5. result += "<book_name>" + book.getBookName() + "</book_name>\n";
  6. result += "<pages>" + book.getPages() + "</pages>\n";
  7. result += "<price>" + book.getPrice() + "</price>\n";
  8. result += "<author>" + book.getAuthor() + "</author>\n";
  9. result += "<isbn>" + book.getIsbn() + "</isbn>\n";
  10. return result;
  11. }
  12. }

定义类JSONFormatter继承自Formatter,里面加入处理JSON格式的具体逻辑:

  1. public class JSONFormatter extends Formatter {
  2. @Override
  3. protected String formating(Book book) {
  4. String result = "";
  5. result += "{\n";
  6. result += "\"book_name\" : \"" + book.getBookName() + "\",\n";
  7. result += "\"pages\" : \"" + book.getPages() + "\",\n";
  8. result += "\"price\" : \"" + book.getPrice() + "\",\n";
  9. result += "\"author\" : \"" + book.getAuthor() + "\",\n";
  10. result += "\"isbn\" : \"" + book.getIsbn() + "\",\n";
  11. result += "}";
  12. return result;
  13. }
  14. }

最后调用代码如下:

  1. public class Test {
  2. public static void main(String[] args) throws Exception {
  3. Book book = new Book();
  4. book.setBookName("Thinking in Java");
  5. book.setPages(880);
  6. book.setPrice(68);
  7. book.setAuthor("Bruce Eckel");
  8. book.setIsbn("9787111213826");
  9. XMLFormatter xmlFormatter = new XMLFormatter();
  10. String result = xmlFormatter.formatBook(book);
  11. System.out.println(result);
  12. JSONFormatter jsonFormatter = new JSONFormatter();
  13. result = jsonFormatter.formatBook(book);
  14. System.out.println(result);
  15. }
  16. }

运行之后,你会发现运行结果和修改前代码的运行结果完全相同。但是使用模板方法之后,代码的可读性有了很大的提高,因为处理格式转换的代码都放到了各自的类当中,而不是全部塞进一个方法中。并且在扩展性上也有了很大的提升,比如你开始感兴趣项目经理说的YAML格式了。

定义类YAMLFormatter继承自Formatter,里面加入处理YAML格式的具体逻辑:

  1. public class YAMLFormatter extends Formatter {
  2. @Override
  3. protected String formating(Book book) {
  4. String result = "";
  5. result += "book_name: " + book.getBookName() + "\n";
  6. result += "pages: " + book.getPages() + "\n";
  7. result += "price: " + book.getPrice() + "\n";
  8. result += "author: " + book.getAuthor() + "\n";
  9. result += "isbn: " + book.getIsbn() + "\n";
  10. return result;
  11. }
  12. }

调用代码只需要加入:

  1. YAMLFormatter yamlFormatter = new YAMLFormatter();
  2. String result = yamlFormatter.formatBook(book);
  3. System.out.println(result);

好了,令人头疼的YAML格式就这样被支持了,只需要在调用的时候决定是实例化XMLFormatter,JSONFormatter还是YAMLFormatter,就可以按照相应的规格进行格式转换了。而且整体的代码很有条理,看起来也很舒心。这个时候,你会轻松地向项目经理调侃一句,还有需要支持的格式吗?

模板方法: 定义一个操作中的算法的骨架,而将一些步骤延迟到子类中,模板方法使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤。

Java设计模式之 — 模板方法(Template Method)的更多相关文章

  1. 宋宝华:Linux设备驱动框架里的设计模式之——模板方法(Template Method)

    本文系转载,著作权归作者所有.商业转载请联系作者获得授权,非商业转载请注明出处. 作者: 宋宝华 来源: 微信公众号linux阅码场(id: linuxdev) 前言 <设计模式>这本经典 ...

  2. 行为型设计模式之模板方法(TEMPLATE METHOD)模式 ,策略(Strategy )模式

    1 模板方法(TEMPLATE METHOD)模式: 模板方法模式把我们不知道具体实现的步聚封装成抽象方法,提供一些按正确顺序调用它们的具体方法(这些具体方法统称为模板方法),这样构成一个抽象基类.子 ...

  3. 设计模式二--模板方法Template method

    模式分类: 书籍推荐:重构-改善既有代码的设计 重构获得模式 设计模式:现代软件设计的特征是"需求的频繁变化".设计模式的要点是 "寻找变化点,然后在变化点处应用设计模式 ...

  4. 设计模式之---模板方法template method的使用

    在面向对象系统的分析与设计过程中经常会遇到这样一种情况:对于某一个业务逻辑(算法实现)在不同的对象中有不同的细节实现,但是逻辑(算法)的框架(或通用的应用算法)是相同的.Template Method ...

  5. 设计模式之模板方法(Template Method)

    在整理模板方法之前,先来说点废话吧.除了记录学习总结,也来记录一下生活吧. 我们公司的老板在北京,老板也会因为项目来公司,不过不是天天来.公司有个同事,只要老板不在就天天迟到,而且一天比一天晚,经常来 ...

  6. 设计模式六: 模板方法(Template Method)

    简介 模板方法属于行为型模式的一种. 实现层面上, 在抽象类中定义了算法或流程的骨架, 将其中易变的部分延迟到子类实现, 也就是允许它的子类实现其中的某些步骤. 模板方法适用于算法不变, 但算法中某些 ...

  7. 行为型设计模式之模板方法(Template Method)

    结构 意图 定义一个操作中的算法的骨架,而将一些步骤延迟到子类中.Te m p l a t e M e t h o d 使得子类可以不改变一个算法的结构即可重定义该算法的某些特定步骤. 适用性 一次性 ...

  8. 一天一个设计模式——模板方法(Template Method)模式

    一.模式说明 现实世界中的模板是用于将事物的结构规律予以固定化.标准化的成果,它体现了结构形式的标准化.例如镂空文字印刷的模板,通过某个模板印刷出来的文字字体大小都是一模一样,但是具体使用什么材质的颜 ...

  9. 折腾Java设计模式之模板方法模式

    博客原文地址:折腾Java设计模式之模板方法模式 模板方法模式 Define the skeleton of an algorithm in an operation, deferring some ...

随机推荐

  1. 四大传值详解:属性传值,单例传值,代理传值,block传值

    一:属性传值 传值情景:从前一个页面向后一个页面传值 a.在后一个页面,根据传值类型和个数,写属性 b.在前一个页面, 为属性赋值 c.在后一个页面, 使用值 例如: 第一个视图: #import & ...

  2. HDU 5538/ 2015长春区域 L.House Building 水题

    题意:求给出图的表面积,不包括底面 #include<bits/stdc++.h> using namespace std ; typedef long long ll; #define ...

  3. 解析Qt元对象系统(五) Q_INVOKABLE与invokeMethod(automatic connection从Qt4.8开始的解释已经与之前不同,发送对象驻足于哪一个线程并不重要,起到决定作用的是接收者对象所驻足的线程以及发射信号(该信号与接受者连接)的线程是不是在同一个线程)good

    概述查看Qt源码可知,Q_INVOKABLE是个空宏,目的在于让moc识别. 使用Q_INVOKABLE来修饰成员函数,目的在于被修饰的成员函数能够被元对象系统所唤起. Q_INVOKABLE与QMe ...

  4. jquerymobile之collapsible可折叠块标题内容动态显示

    jquery mobile提供了一种可折叠的组件--data-role="collapsible",这种组件可以通过点击折叠块头部来展开/折叠块内的内容,详细组件说明可参考w3cs ...

  5. golang import all 类似python import * 效果

    import "io/ioutil" func main() { content, err = iotuil.ReadFile("somefile.txt") ...

  6. bzoj1116 [POI2008]CLO——并查集找环

    题目:https://www.lydsy.com/JudgeOnline/problem.php?id=1116 分析性质,只要有环,那么给环定一下向就满足了条件: 环上点的其他边可以指向外面,所以两 ...

  7. Java图形用户界面编程

    1.Java图形用户界面编程概述 JavaAPI中提供了两套组件用于支持编写图形用户界面:AWT(抽象窗口包)和Swing 2.  容器(Container):重量级容器和轻量级容器(一个容器可以放置 ...

  8. openStack 主机流量计运行状态 随笔记录

    root@ruiy-controller:~# ifconfigeth0      Link encap:Ethernet  HWaddr 0c:c4:7a:0d:97:2c          ine ...

  9. Coursera Algorithms week3 归并排序 练习测验: Counting inversions

    题目原文: An inversion in an array a[] is a pair of entries a[i] and a[j] such that i<j but a[i]>a ...

  10. EasyUI TreeGrid 悬浮效果

    /* 鼠标悬停扩展*/ (function (hasgrid) { if (hasgrid) { /** * 表格提示 */ $.extend($.fn.datagrid.methods, { too ...