java设计模式之组合

听说你们公司最近新推出了一款电子书阅读应用,市场反应很不错,应用里还有图书商城,用户可以在其中随意选购自己喜欢的书籍。你们公司也是对此项目高度重视,加大了投入力度,决定给此应用再增加点功能。

好吧,你也知道你是逃不过此劫了,没过多久你的leader就找到了你。他告诉你目前的应用对每本书的浏览量和销售量做了统计,但现在想增加对每个书籍分类的浏览量和销售量以及所有书籍总的浏览量和销售量做统计的功能,希望你可以来完成这项功能。

领导安排的工作当然是推脱不掉的,你只能硬着头皮上了,不过好在这个功能看起来也不怎么复杂。

你比较喜欢看小说,那么就从小说类的统计功能开始做起吧。首先通过getAllNovels方法可以获取到所有的小说名,然后将小说名传入getBrowseCount方法可以得到该书的浏览量,将小说名传入getSaleCount方法可以得到该书的销售量。你目前只有这几个已知的API可以使用,那么开始动手吧!

[java] view plain copy

  1. public int getNovelsBrowseCount() {
  2. int browseCount = 0;
  3. List<String> allNovels = getAllNovels();
  4. for (String novel : allNovels) {
  5. browseCount += getBrowseCount(novel);
  6. }
  7. return browseCount;
  8. }
  9. public int getNovelsSaleCount() {
  10. int saleCount = 0;
  11. List<String> allNovels = getAllNovels();
  12. for (String novel : allNovels) {
  13. saleCount += getSaleCount(novel);
  14. }
  15. return saleCount;
  16. }

很快你就写下了以上两个方法,这两个方法都是通过获取到所有的小说名,然后一一计算每本小说的浏览量和销售量,最后将结果相加得到总量。

小说类的统计就完成了,然后你开始做计算机类书籍的统计功能,代码如下所示:

[java] view plain copy

  1. public int getComputerBooksBrowseCount() {
  2. int browseCount = 0;
  3. List<String> allComputerBooks = getAllComputerBooks();
  4. for (String computerBook : allComputerBooks) {
  5. browseCount += getBrowseCount(computerBook);
  6. }
  7. return browseCount;
  8. }
  9. public int getComputerBooksSaleCount() {
  10. int saleCount = 0;
  11. List<String> allComputerBooks = getAllComputerBooks();
  12. for (String computerBook : allComputerBooks) {
  13. saleCount += getSaleCount(computerBook);
  14. }
  15. return saleCount;
  16. }

除了使用了getAllComputerBooks方法获取到所有的计算机类书名,其它的代码基本和小说统计中的是一样的。

现在你才完成了两类书籍的统计功能,后面还有医学类、自然类、历史类、法律类、政治类、哲学类、旅游类、美食类等等等等书籍。你突然意识到了一些问题的严重性,工作量大倒还不算什么,但再这么写下去,你的方法就要爆炸了,这么多的方法让人看都看不过来,别提怎么使用了。

这个时候你只好向你的leader求助了,跟他说明了你的困惑。只见你的leader思考了片刻,然后自信地告诉你,使用组合模式不仅可以轻松消除你的困惑,还能出色地完成功能。

他立刻向你秀起了编码操作,首先定义一个Statistics接口,里面有两个待实现方法:

[java] view plain copy

  1. public interface Statistics {
  2. int getBrowseCount();
  3. int getSalesCount();
  4. }

然后定义一个用于统计小说类书籍的NovelStatistics类,实现接口中定义的两个方法:

[java] view plain copy

  1. public class NovelStatistics implements Statistics {
  2. @Override
  3. public int getBrowseCount() {
  4. int browseCount = 0;
  5. List<String> allNovels = getAllNovels();
  6. for (String novel : allNovels) {
  7. browseCount += getBrowseCount(novel);
  8. }
  9. return browseCount;
  10. }
  11. @Override
  12. public int getSalesCount() {
  13. int saleCount = 0;
  14. List<String> allNovels = getAllNovels();
  15. for (String novel : allNovels) {
  16. saleCount += getSaleCount(novel);
  17. }
  18. return saleCount;
  19. }
  20. }

在这两个方法中分别统计了小说类书籍的浏览量和销售量。那么同样的方法,你的leader又定义了一个ComputerBookStatistics类用于统计计算机类书籍的浏览量和销售量:

[java] view plain copy

  1. public class ComputerBookStatistics implements Statistics {
  2. @Override
  3. public int getBrowseCount() {
  4. int browseCount = 0;
  5. List<String> allComputerBooks = getAllComputerBooks();
  6. for (String computerBook : allComputerBooks) {
  7. browseCount += getBrowseCount(computerBook);
  8. }
  9. return browseCount;
  10. }
  11. @Override
  12. public int getSalesCount() {
  13. int saleCount = 0;
  14. List<String> allComputerBooks = getAllComputerBooks();
  15. for (String computerBook : allComputerBooks) {
  16. saleCount += getSaleCount(computerBook);
  17. }
  18. return saleCount;
  19. }
  20. }

这样将具体的统计实现分散在各个类中,就不会再出现你刚刚那种方法爆炸的情况了。不过这还没开始真正使用组合模式呢,好戏还在后头,你的leader吹嘘道。

再定义一个MedicalBookStatistics类实现了Statistics接口,用于统计医学类书籍的浏览量和销售量,代码如下如示:

[java] view plain copy

  1. public class MedicalBookStatistics implements Statistics {
  2. @Override
  3. public int getBrowseCount() {
  4. int browseCount = 0;
  5. List<String> allMedicalBooks = getAllMedicalBooks();
  6. for (String medicalBook : allMedicalBooks) {
  7. browseCount += getBrowseCount(medicalBook);
  8. }
  9. return browseCount;
  10. }
  11. @Override
  12. public int getSalesCount() {
  13. int saleCount = 0;
  14. List<String> allMedicalBooks = getAllMedicalBooks();
  15. for (String medicalBook : allMedicalBooks) {
  16. saleCount += getSaleCount(medicalBook);
  17. }
  18. return saleCount;
  19. }
  20. }

不知道你发现了没有,计算机类书籍和医学类书籍其实都算是科技类书籍,它们是可以组合在一起的。这个时候你的leader定义了一个TechnicalStatistics类用于对科技这一组合类书籍进行统计:

[java] view plain copy

  1. public class TechnicalStatistics implements Statistics {
  2. private List<Statistics> statistics = new ArrayList<Statistics>();
  3. public TechnicalStatistics() {
  4. statistics.add(new ComputerBookStatistics());
  5. statistics.add(new MedicalBookStatistics());
  6. }
  7. @Override
  8. public int getBrowseCount() {
  9. int browseCount = 0;
  10. for (Statistics s : statistics) {
  11. browseCount += s.getBrowseCount();
  12. }
  13. return browseCount;
  14. }
  15. @Override
  16. public int getSalesCount() {
  17. int saleCount = 0;
  18. for (Statistics s : statistics) {
  19. saleCount += s.getBrowseCount();
  20. }
  21. return saleCount;
  22. }
  23. }

可以看到,由于这个类是组合类,和前面几个类还是有不少区别的。首先TechnicalStatistics中有一个构造函数,在构造函数中将计算机类书籍和医学类书籍作为子分类添加到statistics列表当中,然后分别在getBrowseCount和getSalesCount方法中遍历所有的子分类,计算出它们各自的浏览量和销售量,然后相加得到总额返回。

组合模式的扩展性非常好,没有各种条条框框,想怎么组合就怎么组合,比如所有书籍就是由各个分类组合而来的,你的leader马上又向你炫耀了统计所有书籍的浏览量和销售量的办法。

定义一个AllStatistics类实现了Statistics接口,具体代码如下所示:

[java] view plain copy

  1. public class AllStatistics implements Statistics {
  2. private List<Statistics> statistics = new ArrayList<Statistics>();
  3. public AllStatistics() {
  4. statistics.add(new NovelStatistics());
  5. statistics.add(new TechnicalStatistics());
  6. }
  7. @Override
  8. public int getBrowseCount() {
  9. int browseCount = 0;
  10. for (Statistics s : statistics) {
  11. browseCount += s.getBrowseCount();
  12. }
  13. return browseCount;
  14. }
  15. @Override
  16. public int getSalesCount() {
  17. int saleCount = 0;
  18. for (Statistics s : statistics) {
  19. saleCount += s.getBrowseCount();
  20. }
  21. return saleCount;
  22. }
  23. }

在AllStatistics的构造函数中将小说类书籍和科技类书籍作为子分类添加到了statistics列表当中,目前你也就只写好了这几个分类。然后使用同样的方法在getBrowseCount和getSalesCount方法中统计出所有书籍的浏览量和销售量。

当前组合结构的示意图如下:

现在你就可以非常方便的得到任何分类书籍的浏览量和销售量了,比如说获取科技类书籍的浏览量,你只需要调用:

[java] view plain copy

  1. new TechnicalStatistics().getBrowseCount();

而获取所有书籍的总销量,你只需要调用:

[java] view plain copy

  1. new AllStatistics().getSalesCount();

当然你后面还可以对这个组合结构随意地改变,添加各种子分类书籍,而且子分类的层次结构可以任意深,正如前面所说,组合模式的扩展性非常好。

你的leader告诉你,目前他写的这份代码重复度比较高,其实还可以好好优化一下的,把冗余代码都去除掉。当然这个任务就交给你来做了,你的leader可是大忙人,早就一溜烟跑开了。

组合:将对象组合成树形结构以表示“部分-整体”的层次结构。组合模式使得用户对单个对象和组合对象的使用具有一致性。

时间: 2024-10-21 05:35:51

java设计模式之组合的相关文章

图解Java设计模式之组合模式

图解Java设计模式之组合模式 看一个学校院系展示需求 传统方案解决学院院系展示(类图) 传统方案解决学校院系展示存在的问题分析 组合模式基本介绍 组合模式的原理类图 组合模式解决学校院系展示的应用实例 组合模式在JDK集合的源码分析 组合模式的注意事项和细节 看一个学校院系展示需求 编写程序展示一个学校院系结构 :需求是这样,要在一个页面中展示出学校的院系组成,一个学校有多个学院,一个学院有多个系.如图 : 传统方案解决学院院系展示(类图) 传统方案解决学校院系展示存在的问题分析 1)将学院看

java设计模式之组合模式

树形结构在软件中随处可见,例如操作系统中的目录结构.应用软件中的菜单.办公系统中的公司组织结构等等,如何运用面向对象的方式来处理这种树形结构是组合模式需要解决的问题,组合模式通过一种巧妙的设计方案使得用户可以一致性地处理整个树形结构或者树形结构的一部分,也可以一致性地处理树形结构中的叶子节点(不包含子节点的节点)和容器节点(包含子节点的节点).下面将学习这种用于处理树形结构的组合模式. 11.1 设计杀毒软件的框架结构 Sunny软件公司欲开发一个杀毒(AntiVirus)软件,该软件既可以对某

Java设计模式应用——组合模式

组合模式实际上是一种树形数据结构.以windows目录系统举例,怎么样用java语言描述一个文件夹? 定义一个文件夹类,文件夹类中包含若干个子文件类和若干个文件类. 进一步抽象,把文件夹和文件都看做节点,于是一个文件夹就可以描述为一个节点类,包含若干个子节点. 我们看看组合模式的代码 // 抽象节点 public abstract class Node { protected String name; abstract void add(Node node); abstract void rem

浅谈JAVA设计模式之——组合模式(Composite)

转载请注明出处:http://blog.csdn.net/l1028386804/article/details/45458081 一.概述 将对象组合成树形结构以表示"部分-整体"的层次结构."Composite使得用户对单个对象和组合对象的使用具有一致性. 二.适用性 1.你想表示对象的部分-整体层次结构. 2.你希望用户忽略组合对象与单个对象的不同,用户将统一地使用组合结构中的所有对象. 三.参与者 1.Component 为组合中的对象声明接口. 在适当的情况下,实现

java设计模式5.组合模式、门面模式、享元模式、桥接模式

组合模式 在面向对象的语言中,树结构有着巨大的威力,一个基于继承的类型的等级结构便是一个数结构,一个基于合成的对象结构也是一个数结构.组合模式将部分与整体的关系用树结构表示出来,使得客户端把一个个单独的成分对象和由它们组合而成的合成对象同等看待. 抽象构建角色:一个抽象角色,给参加组合的对象规定一个接口,这个接口给出共有的接口及默认行为. 树叶构建角色:代表参加组合的树叶对象,没有子对象,定义参加组合的原始对象行为. 树枝构建角色:代表参加组合的有子对象的对象,并给出树枝构件对象的行为. 对于子

java设计模式_组合模式

/**  *   */ package com.wangbiao.design.composite; import java.util.ArrayList; import java.util.List; /**  * @Title: Component.java  * @Package com.wangbiao.design.composite  * @Description: TODO   * @author wangbiao     * @date 2014-9-28 上午9:50:26 

JAVA设计模式:组合模式

组合(总体与部分关系)模式:将不同可是相关的对象组合成树形结构以实现"部分-总体"的层次结构,使得用户对单个对象和组合对象的使用具有一致性. * 模式角色组成: * 1.Component对象: 是组合中的对象接口,是全部类共同拥有的接口.是用于统一定义总体中的部分. 2.Leaf对象: 总体中的部分,没有下一级. 3.Composite对象: 用来存储子部件.在Component接口中实现与部分有关操作. 以公司构成为例,比方公司有总公司.总公司下有分公司,分公司下有办事处等.构成一

理解设计模式(组合)

组合模式,保证系统中相同类的组合对象和单一对象是一致的. 特定环境 若干对象树形架构,具有**整体——部分”层次结构 其中,操作任意若干对象的组合结构,如同操作单个对象 一类问题 整理和部分如何被一致的对待 解决方案 组合模式有3个角色: 组件接口 Component :规范类行为,一般增删叶节点或组合节点,由接口或抽象类实现 叶节点 Leaf :增删对该角色无效,实现Componet, 一般有具体子类实现 组合节点 Composite :封装叶节点引用集,实现增删逻辑,实现Componet,

Java 设计模式系列(九)组合模式

Java 设计模式系列(九)组合模式 将对象组合成树形结构以表示"部分-整体"的层次结构.组合模式使得用户对单个对象的使用具有一致性. 一.组合模式结构 Component: 抽象的组件对象,为组合中的对象声明接口,让客户端可以通过这个接口来访问和管理整个对象结构,可以在里面为定义的功能提供缺省的实现. Leaf: 叶子节点对象,定义和实现叶子对象的行为,不再包含其它的子节点对象. Composite: 组合对象,通常会存储子组件,定义包含子组件的那些组件的行为,并实现在组件接口中定义