java设计模式之组合
听说你们公司最近新推出了一款电子书阅读应用,市场反应很不错,应用里还有图书商城,用户可以在其中随意选购自己喜欢的书籍。你们公司也是对此项目高度重视,加大了投入力度,决定给此应用再增加点功能。
好吧,你也知道你是逃不过此劫了,没过多久你的leader就找到了你。他告诉你目前的应用对每本书的浏览量和销售量做了统计,但现在想增加对每个书籍分类的浏览量和销售量以及所有书籍总的浏览量和销售量做统计的功能,希望你可以来完成这项功能。
领导安排的工作当然是推脱不掉的,你只能硬着头皮上了,不过好在这个功能看起来也不怎么复杂。
你比较喜欢看小说,那么就从小说类的统计功能开始做起吧。首先通过getAllNovels方法可以获取到所有的小说名,然后将小说名传入getBrowseCount方法可以得到该书的浏览量,将小说名传入getSaleCount方法可以得到该书的销售量。你目前只有这几个已知的API可以使用,那么开始动手吧!
- public int getNovelsBrowseCount() {
- int browseCount = 0;
- List<String> allNovels = getAllNovels();
- for (String novel : allNovels) {
- browseCount += getBrowseCount(novel);
- }
- return browseCount;
- }
- public int getNovelsSaleCount() {
- int saleCount = 0;
- List<String> allNovels = getAllNovels();
- for (String novel : allNovels) {
- saleCount += getSaleCount(novel);
- }
- return saleCount;
- }
很快你就写下了以上两个方法,这两个方法都是通过获取到所有的小说名,然后一一计算每本小说的浏览量和销售量,最后将结果相加得到总量。
小说类的统计就完成了,然后你开始做计算机类书籍的统计功能,代码如下所示:
- public int getComputerBooksBrowseCount() {
- int browseCount = 0;
- List<String> allComputerBooks = getAllComputerBooks();
- for (String computerBook : allComputerBooks) {
- browseCount += getBrowseCount(computerBook);
- }
- return browseCount;
- }
- public int getComputerBooksSaleCount() {
- int saleCount = 0;
- List<String> allComputerBooks = getAllComputerBooks();
- for (String computerBook : allComputerBooks) {
- saleCount += getSaleCount(computerBook);
- }
- return saleCount;
- }
除了使用了getAllComputerBooks方法获取到所有的计算机类书名,其它的代码基本和小说统计中的是一样的。
现在你才完成了两类书籍的统计功能,后面还有医学类、自然类、历史类、法律类、政治类、哲学类、旅游类、美食类等等等等书籍。你突然意识到了一些问题的严重性,工作量大倒还不算什么,但再这么写下去,你的方法就要爆炸了,这么多的方法让人看都看不过来,别提怎么使用了。
这个时候你只好向你的leader求助了,跟他说明了你的困惑。只见你的leader思考了片刻,然后自信地告诉你,使用组合模式不仅可以轻松消除你的困惑,还能出色地完成功能。
他立刻向你秀起了编码操作,首先定义一个Statistics接口,里面有两个待实现方法:
- public interface Statistics {
- int getBrowseCount();
- int getSalesCount();
- }
然后定义一个用于统计小说类书籍的NovelStatistics类,实现接口中定义的两个方法:
- public class NovelStatistics implements Statistics {
- @Override
- public int getBrowseCount() {
- int browseCount = 0;
- List<String> allNovels = getAllNovels();
- for (String novel : allNovels) {
- browseCount += getBrowseCount(novel);
- }
- return browseCount;
- }
- @Override
- public int getSalesCount() {
- int saleCount = 0;
- List<String> allNovels = getAllNovels();
- for (String novel : allNovels) {
- saleCount += getSaleCount(novel);
- }
- return saleCount;
- }
- }
在这两个方法中分别统计了小说类书籍的浏览量和销售量。那么同样的方法,你的leader又定义了一个ComputerBookStatistics类用于统计计算机类书籍的浏览量和销售量:
- public class ComputerBookStatistics implements Statistics {
- @Override
- public int getBrowseCount() {
- int browseCount = 0;
- List<String> allComputerBooks = getAllComputerBooks();
- for (String computerBook : allComputerBooks) {
- browseCount += getBrowseCount(computerBook);
- }
- return browseCount;
- }
- @Override
- public int getSalesCount() {
- int saleCount = 0;
- List<String> allComputerBooks = getAllComputerBooks();
- for (String computerBook : allComputerBooks) {
- saleCount += getSaleCount(computerBook);
- }
- return saleCount;
- }
- }
这样将具体的统计实现分散在各个类中,就不会再出现你刚刚那种方法爆炸的情况了。不过这还没开始真正使用组合模式呢,好戏还在后头,你的leader吹嘘道。
再定义一个MedicalBookStatistics类实现了Statistics接口,用于统计医学类书籍的浏览量和销售量,代码如下如示:
- public class MedicalBookStatistics implements Statistics {
- @Override
- public int getBrowseCount() {
- int browseCount = 0;
- List<String> allMedicalBooks = getAllMedicalBooks();
- for (String medicalBook : allMedicalBooks) {
- browseCount += getBrowseCount(medicalBook);
- }
- return browseCount;
- }
- @Override
- public int getSalesCount() {
- int saleCount = 0;
- List<String> allMedicalBooks = getAllMedicalBooks();
- for (String medicalBook : allMedicalBooks) {
- saleCount += getSaleCount(medicalBook);
- }
- return saleCount;
- }
- }
不知道你发现了没有,计算机类书籍和医学类书籍其实都算是科技类书籍,它们是可以组合在一起的。这个时候你的leader定义了一个TechnicalStatistics类用于对科技这一组合类书籍进行统计:
- public class TechnicalStatistics implements Statistics {
- private List<Statistics> statistics = new ArrayList<Statistics>();
- public TechnicalStatistics() {
- statistics.add(new ComputerBookStatistics());
- statistics.add(new MedicalBookStatistics());
- }
- @Override
- public int getBrowseCount() {
- int browseCount = 0;
- for (Statistics s : statistics) {
- browseCount += s.getBrowseCount();
- }
- return browseCount;
- }
- @Override
- public int getSalesCount() {
- int saleCount = 0;
- for (Statistics s : statistics) {
- saleCount += s.getBrowseCount();
- }
- return saleCount;
- }
- }
可以看到,由于这个类是组合类,和前面几个类还是有不少区别的。首先TechnicalStatistics中有一个构造函数,在构造函数中将计算机类书籍和医学类书籍作为子分类添加到statistics列表当中,然后分别在getBrowseCount和getSalesCount方法中遍历所有的子分类,计算出它们各自的浏览量和销售量,然后相加得到总额返回。
组合模式的扩展性非常好,没有各种条条框框,想怎么组合就怎么组合,比如所有书籍就是由各个分类组合而来的,你的leader马上又向你炫耀了统计所有书籍的浏览量和销售量的办法。
定义一个AllStatistics类实现了Statistics接口,具体代码如下所示:
- public class AllStatistics implements Statistics {
- private List<Statistics> statistics = new ArrayList<Statistics>();
- public AllStatistics() {
- statistics.add(new NovelStatistics());
- statistics.add(new TechnicalStatistics());
- }
- @Override
- public int getBrowseCount() {
- int browseCount = 0;
- for (Statistics s : statistics) {
- browseCount += s.getBrowseCount();
- }
- return browseCount;
- }
- @Override
- public int getSalesCount() {
- int saleCount = 0;
- for (Statistics s : statistics) {
- saleCount += s.getBrowseCount();
- }
- return saleCount;
- }
- }
在AllStatistics的构造函数中将小说类书籍和科技类书籍作为子分类添加到了statistics列表当中,目前你也就只写好了这几个分类。然后使用同样的方法在getBrowseCount和getSalesCount方法中统计出所有书籍的浏览量和销售量。
当前组合结构的示意图如下:
现在你就可以非常方便的得到任何分类书籍的浏览量和销售量了,比如说获取科技类书籍的浏览量,你只需要调用:
- new TechnicalStatistics().getBrowseCount();
而获取所有书籍的总销量,你只需要调用:
- new AllStatistics().getSalesCount();
当然你后面还可以对这个组合结构随意地改变,添加各种子分类书籍,而且子分类的层次结构可以任意深,正如前面所说,组合模式的扩展性非常好。
你的leader告诉你,目前他写的这份代码重复度比较高,其实还可以好好优化一下的,把冗余代码都去除掉。当然这个任务就交给你来做了,你的leader可是大忙人,早就一溜烟跑开了。
组合:将对象组合成树形结构以表示“部分-整体”的层次结构。组合模式使得用户对单个对象和组合对象的使用具有一致性。
java设计模式之组合的更多相关文章
- java设计模式之组合模式
组合模式 组合模式,将对象组合成树形结构以表示“部分-整体”的层次结构,组合模式使得用户对单个对象和组合对象的使用具有一致性.掌握组合模式的重点是要理解清楚 “部分/整体” 还有 ”单个对象“ 与 & ...
- [译]Java 设计模式之组合
(文章翻译自Java Design Pattern: Composite) 组合模式相对来说是比较简单的,但是它在很多设计中被用到,比如SWT.Eclipse工作空间等等.它创建了一个可以通过唯一的方 ...
- Java设计模式应用——组合模式
组合模式实际上是一种树形数据结构.以windows目录系统举例,怎么样用java语言描述一个文件夹? 定义一个文件夹类,文件夹类中包含若干个子文件类和若干个文件类. 进一步抽象,把文件夹和文件都看做节 ...
- java设计模式5.组合模式、门面模式、享元模式、桥接模式
组合模式 在面向对象的语言中,树结构有着巨大的威力,一个基于继承的类型的等级结构便是一个数结构,一个基于合成的对象结构也是一个数结构.组合模式将部分与整体的关系用树结构表示出来,使得客户端把一个个单独 ...
- JAVA设计模式之组合模式(composite)
组合模式:树状结构专用模式 代码如下: package com.srr.dp.composite; import java.util.ArrayList; import java.util.List; ...
- Java设计模式之 — 组合(Composite)
转载请注明出处:http://blog.csdn.net/guolin_blog/article/details/9153753 听说你们公司最近新推出了一款电子书阅读应用,市场反应很不错,应用里还有 ...
- 老和尚给小和尚讲故事引发了Java设计模式:组合模式
目录 示例 组合模式 定义 意图 主要解决问题 优缺点 安全式和透明式的组合模式 安全式的合成模式的结构 透明式的合成模式的结构 老和尚和小和尚的故事 示例 有一个绘图系统,可以描绘各种图形,假设现在 ...
- JAVA 设计模式 组合模式
用途 组合模式 (Component) 将对象组合成树形结构以表示“部分-整体”的层次结构.组合模式使得用户对单个对象和组合对象的使用具有唯一性. 组合模式是一种结构型模式. 结构
- Java设计模式——组合模式
JAVA 设计模式 组合模式 用途 组合模式 (Component) 将对象组合成树形结构以表示“部分-整体”的层次结构.组合模式使得用户对单个对象和组合对象的使用具有唯一性. 组合模式是一种结构型模 ...
随机推荐
- 微软移除Visual Studio 2015中的UML
微软已经在Visual Studio 2015中移除了UML(Unified Modeling Language,统一建模语言),原因是该语言使用率过低.因此微软要优化产品结构,把好钢用在刀刃上. V ...
- 使用phonegap开发安卓HLS播放软件解决方案
目前使用phonegap开发的手机应用,很少涉及视频播放的功能,究其原因,主要是phonegap提供的API里面对视频播放功能支持度不够,当然播放音频一般情况下还是能够实现的,由于工作需要,自己研究了 ...
- MTK平台源码 mt6589-jb3-x_20131122
韩梦飞沙 韩亚飞 313134555@qq.com yue31313 han_meng_fei_sha
- luogu P1137 旅行计划
题目描述 小明要去一个国家旅游.这个国家有N个城市,编号为1-N,并且有M条道路连接着,小明准备从其中一个城市出发,并只往东走到城市i停止. 所以他就需要选择最先到达的城市,并制定一条路线以城市i为终 ...
- [CodeForces-759D]Bacterial Melee
题目大意: 有一串n个字母,每个位置的字母可以同化边上的一个字母, 比如:ab可以变成aa或者bb. 相对的两个同化不能同时发生,比如ab不能变成ba. 现在给你一个字符串,问你经过任意次数的同化过程 ...
- jquery ajax 的封装
var tooAjaxData = new Object(); tooAjaxData = function () { this.AjaxUrl =" ";}; bookInfoC ...
- convirt介绍
convirt2.0是一款使用python和jquery结合编写的其于web的集中管理xen服务的程序.该程序在xen 社区项目,管理项目中被使用的量很高,convirt开发有开源版本与企业版本,企业 ...
- Linux PHP 编译参数详解(一)
Fast-CGI: ./configure --prefix=/usr/local/php --enable-fastcgi --enable-force-cgi-redirect --with-co ...
- JavaScript里的循环方法:forEach,for-in,for-of
JavaScript诞生已经有20多年了,我们一直使用的用来循环一个数组的方法是这样的: for (var index = 0; index < myArray.length; index++) ...
- [转] Cz/C++中栈空间、堆空间,及内存区域的划分
kevinGao, 原文地址 一个由C/C++编译的程序占用的内存分为以下几个部分: 1.栈区(stack):又编译器自动分配释放,存放函数的参数值,局部变量的值等,其操作方式类似于数据结构的栈. 2 ...