Head First 设计模式 —— 10. 迭代器 (Iterator) 模式
思考题
public void printMenu() {
PancakeHouseMenu pancakeHouseMenu = new PancakeHouseMenu();
ArrayList breakfastItems = pancakeHouseMenu.getMenuItems();
DinerMenu dinerMenu = new DinerMenu();
MenuItem[] lunchItems = dinerMenu.getMenuItems();
for (int i = 0; i < breakfastItems.size(); ++i) {
MenuItem menuItem = (MenuItem) breakfastItems.get(i);
System.out.print(menuItem.getName() + " ");
System.out.println(menuItem.getPrice() + " ");
System.out.println(menuItem.getDescription());
}
for (int i = 0; i < lunchItems.length; ++i) {
MenuItem menuItem = lunchItems[i];
System.out.print(menuItem.getName() + " ");
System.out.println(menuItem.getPrice() + " ");
System.out.println(menuItem.getDescription());
}
}
根据我们的 printMenu() 实现,下列哪一项为真? P322
- [x] A. 我们是针对
PancakeHouseMenu和DinerMenu的具体实现编码,而不是针对接口。- 针对具体实现编程,没有解耦,无法动态替换
- [ ] B. 女招待没有实现 Java 招待 API ,所以她没有遵守标准。
- 女招待有一个招待规格,算是实现了一个隐式 API,并且内部实现封装了变化
- [x] C. 如果我们决定从
DinerMenu切换到另一种菜单,此菜单的项是用Hashtable来存放的,我们会因此需要修改女招待中的许多代码。Hashtable是一种新的具体实现,需要修改代码进行适配
- [x] D. 女招待需要知道每个菜单如何表达内部的菜单项集合,这违反了封装。
- 女招待的实现依赖菜单内部实现,与菜单强耦合,没有封装变化
- [x] E. 我们有重复的代码;
printMenu()方法需要两个循环,来遍历两种不同的菜单。如果我们加上第三种菜单,我们就需要第三个循环。- 由于是针对菜单项集合的具体实现编程,所以增加新菜单就需要增加循环
- [ ] F. 这个实现并没有基于 MXML(Menu XML),所以就没有办法互操作。
The implementation isn't based on MXML(Menu XML) and so isn't as interoperable as it shoule be.
互操作性(英文:Interoperability;中文又称为:协同工作能力,互用性)作为一种特性,它指的是不同的系统和组织机构之间相互合作,协同工作(即互操作)的能力。就软件而言,互操作性——这条术语用来描述的是不同的程序(programs)借助于同一套交换格式(exchange formats)来交换数据,读写相同文件格式(file formats)以及采用相同协议(protocols)的能力。
刚开始不懂,看了维基百科后的解释后有点似懂非懂。这个实现内部直接处理了不同数据格式,具有一定的互操作性。
思考题
请继续完成 PancakeHouseIterator 的实现,并对 PancakeHouseMenu 类做出必要的修改。 P327
public class PancakeHouseIterator implements Iterator {
ArrayList items;
int position = 0;
public PancakeHouseIterator(ArrayList items) {
this.items = items;
}
public Object next() {
return items.get(position++);
}
public boolean hasNext() {
return position >= items.size();
}
}
public class PancakeHouseMenu {
ArrayList menuItems;
// 构造器和其他方法
// 移除 public ArrayList getMenuItems() 方法
public Iterator createIterator() {
return new PancakeHouseIterator(menuItems);
}
}
迭代器模式
提供一种方法顺序访问一个聚合对象中的各个元素,而又不暴露其内部的表示。 P336

特点
- 遍历集合内部元素而不暴露内部表示
P336 - 让聚合的接口和实现变得简洁
P336 - 管理对象集合,不必理会遍历的事情
P336
思考题
迭代器模式的类图看起来很像我们所学过的另一个模式;你知道是哪个模式吗?提示:子类决定要创建哪个对象。 P337
- 工厂方法模式
设计原则
单一职责原则:一个类应该只有一个引起变化的原因 P339
- 高内聚:一个模块或一个类只支持一组相关的功能
P339
特点
- 尽量将一个责任只指派给一个类
P339 - 遵循单一职责原则的类是高内聚的
P339
思考题
请很快写下为了能让咖啡厅的菜单代码符合我们的框架,我们要对它做的三件事情: P342
public class CafeMenu {
HashTable menuItems = new Hashtable();
public CafeMenu() {
// 省略下述参数
// addItem(...);
// addItem(...);
// addItem(...);
}
public void addItem(String name, String description, boolean vegetarian, double price) {
MenuItem menuItem = new MenuItem(name, description, vegetarian, price);
menuItems.put(menuItem.getName(), menuItem);
}
public Hashtable getItems() {
return menuItems;
}
}
- 实现
Menu接口 - 删除
getItems()方法 - 增加
createIterator()方法
本文首发于公众号:满赋诸机(点击查看原文) 开源在 GitHub :reading-notes/head-first-design-patterns
Head First 设计模式 —— 10. 迭代器 (Iterator) 模式的更多相关文章
- 1、迭代器 Iterator模式 一个一个遍历 行为型设计模式
1.Iterator模式 迭代器(iterator)有时又称游标(cursor)是程序设计的软件设计模式,可在容器(container,例如链表或者阵列)上遍访的接口,设计人员无需关心容器的内容. I ...
- 【设计模式大法】Iterator模式
Iterator模式 --一个一个遍历 在Java中的for语句中 i++的作用是让 i 的值在每次循环后自增1,这样就可以访问数组中的下一个元素.下下一个元素.再下下一个元素,也就实现了从头至尾逐一 ...
- Java 实现迭代器(Iterator)模式
类图 /** * 自己定义集合接口, 相似java.util.Collection * 用于数据存储 * @author stone * */ public interface ICollection ...
- 设计模式—迭代器Iterator模式
什么是迭代器模式? 让用户通过特定的接口访问容器的数据,不需要了解容器内部的数据结构. 首先我们先模仿集合中ArrayList和LinkedList的实现.一个是基于数组的实现.一个是基于链表的实现, ...
- 设计模式C++描述----20.迭代器(Iterator)模式
一. 举例说明 我们知道,在 STL 里提供 Iterator 来遍历 Vector 或者 List 数据结构. Iterator 模式也正是用来解决对一个聚合对象的遍历问题,将对聚合的遍历封装到一个 ...
- 设计模式——迭代器(Iterator)模式
概述 迭代器模式简单的说(按我目前的理解)就是一个类提供一个对外迭代的接口,方面调用者迭代.这个迭代接口至少包括两个方法:hasNext()--用于判断是否还有下一个,next()--用于取出下一个对 ...
- GoLang设计模式10 - 中介者模式
中介者模式是一种行为型设计模式.在中介者模式中创建了一个中介对象来负责不同类间的通信.因为这些类不需要直接交互,所以也就能避免它们之间的直接依赖,实现解耦的效果. 中介者模式的一个典型案例是老式小火车 ...
- JS常用的设计模式(10)——模版方法模式
模式方法是预先定义一组算法,先把算法的不变部分抽象到父类,再将另外一些可变的步骤延迟到子类去实现.听起来有点像工厂模式( 非前面说过的简单工厂模式 ). 最大的区别是,工厂模式的意图是根据子类的实现最 ...
- 设计模式(10)--Facade(外观模式)--结构型
作者QQ:1095737364 QQ群:123300273 欢迎加入! 1.模式定义: 外观模式提供了一个统一的接口,用来访问子系统中的一群接口.外观定义了一个高层接口,让子系统更容易使 ...
随机推荐
- js 几种跨域解决方法
同源策略: JS只能与同一个域中的页面进行通讯,必须是协议.域名.端口都相同,相同域下才能相互通信,这可以被认为是一种通信原则,叫同源策略. 跨域: js跨域是指通过js在不同的域之间进行数据传输或通 ...
- NOI Online #1 入门组 魔法
全网都是矩阵快速幂,我只会倍增DP 其实这题与 AcWing 345. 牛站 还是比较像的,那题可以矩阵快速幂 / 倍增,这题也行. 先 \(Floyd\) 预处理两点之间不用魔法最短距离 \(d_{ ...
- 算法(图论)——最小生成树及其题目应用(prim和Kruskal算法实现)
题目 n个村庄间架设通信线路,每个村庄间的距离不同,如何架设最节省开销? Kruskal算法 特点 适用于稀疏图,时间复杂度 是nlogn的. 核心思想 从小到大选取不会产生环的边. 代码实现 代码中 ...
- [日常摸鱼]poj2778 DNA Sequence
这题太神啦 题意:求长度为$n$的不包含给定DNA序列的DNA序列个数,给定的不超过10个 构建出Trie图,用$danger[i]$来表示不能走到$i$,对于DNA序列结尾的结点$danger$设为 ...
- 《Spring Boot 实战纪实》之关键点文档
目录 前言 (思维篇)人人都是产品经理 1.需求文档 1.1 需求管理 1.2 如何攥写需求文档 1.3 需求关键点文档 2 原型设计 2.1 缺失的逻辑 2.2 让想法跃然纸上 3 开发设计文档 3 ...
- Django + FastDFS (分布式远程服务器存储文件)
之前随笔过一篇Docker来搭建分布式文件系统FastDfs就跳过了 https://www.cnblogs.com/xcsg/p/10901461.html FastDFS的Python (dja ...
- python线性回归
一.理论基础 1.回归公式 对于单元的线性回归,我们有:f(x) = kx + b 的方程(k代表权重,b代表截距). 对于多元线性回归,我们有: 或者为了简化,干脆将b视为k0·x0,,其中k0为1 ...
- 1.docker介绍、命令、容器、镜像、数据卷、Dockerfile、常用软件安装、推送阿里云
一.docker介绍 1.docker是什么 一款产品从开发到上线,从操作系统,到运行环境,再到应用配置.作为开发+运维之间的协作我们需要关心很多东西,这也是很多互联网公司都不得不面对的问题,特别是各 ...
- python初学者-计算小于100的最大素数
for n in range(100,1,-1): for i in range(2,n): if n%i==0: break else: print(n,end=' ')
- 前端可视化开发--liveload
在前端开发中,我们会频繁的修改html.css.js,然后刷新页面,开效果,再调整,再刷新,不知不觉会浪费掉我们很多时间.有没有什么方法,我在编辑器里面改了代码以后,只要保存,浏览器就能实时刷新.经过 ...
