迭代器模式

基本理解

  • 迭代器模式(Iterrator):提供一个方法顺序访问一个聚合对象中的各个元素,而又不暴露该元素的内部表示。
  • 当你访问一个聚合对象,而且不管这些对象是什么都需要遍历的时候,你就应该考虑用迭代器模式。
  • 你需要对聚集有多种方式遍历时,可以考虑用迭代器模式。
  • 迭代器模式就是分离了集合对象的遍历行为,抽象出一个迭代器类来负责,这样既可以做到不暴露集合的内部结构,又可让外部代码透明地访问集合内部的数据。
  • 迭代器定义了一个用于访问集合元素并记录当前元素的接口。
  • 不同的迭代器可以执行不同的迭代策略。
  • 外部迭代器和内部迭代器
    • 外部迭代器
    • 外部迭代器让客户端直接操作迭代过程,所以客户端需要知道外部迭代器才能使用。但是它为客户端提供了更多的控制
    • 客户端创建并维护外部迭代器
    • 客户端可以使用不同外部迭代器实现多种类型的遍历
    • 内部迭代器
      • 客户端不需要知道任何外部迭代器,而是可以通过集合对象的特殊接口,或者一次访问一个元素,或者向集合中的每个元素发送消息。
      • 集合对象本身创建并维护它的外部迭代器
      • 集合对象可以在不修改客户端代码的情况下,选择不同的外部迭代器

何时使用迭代器

  • 需要访问组合对象的内容,而又不是暴露其内部标示
  • 需要通过多种方式遍历组合对象
  • 需要提供一个统一的接口,用来遍历各种类型的组合对象

在Cocoa Touch框架中使用迭代器模式

  • 苹果公司用自己命名规则“枚举器/枚举”改写了迭代器模式,用于相关基础类的各种方法。也就是说枚举就是苹果版本的迭代。

  • 基础框架中的NSEnumerator类实现了迭代器模式。抽象NSEnumerator类的私有具体子类返回枚举器对象,能够顺序遍历各种几何--数组、集(set)、字典(值与键),把集合的对象返回给客户端。

  • NSEnumerator可以枚举NSArray/NSDictionary和NSSet对象中的元素。它本身是个抽象类。它依靠几个工厂方法,如objectEnumerator或keyEnumerator,来创建并返回相应的具体枚举器对象。

  • 从iOS4开始,有了另一种枚举Cocoa Touch框架中集合对象的方法,叫做基于块的枚举。

  • 快速枚举,这个是苹果推荐的枚举方法。它允许把集合对象的枚举直接用作for循环的一部分,无需使用其他枚举器对象,而且比传统的基于索引的for循环效率更高,形如:

     NSArray *arr = @[@"zhangsan",@"lisi",@"wangwu"];
    
     for(NSString *item in arr)
    {
    NSLog(@"%@",item);
    }

例子

该例子定义了一个集合Set和一个迭代器ConcreteIterator。然后通过迭代器来输出集合Set对象的元素。

ConcreteSet.h

//
// ConcreteSet.h
// IterationDemo
//
// Created by zhanggui on 15/8/8.
// Copyright (c) 2015年 zhanggui. All rights reserved.
// #import <Foundation/Foundation.h> @interface ConcreteSet : NSObject
{
NSMutableArray *items;
}
-(NSInteger)getCount;
-(id)getItemFromIndex:(NSInteger)index; -(void)insertItem:(id)item; @end

ConcreteSet.m

     	//
// ConcreteSet.m
// IterationDemo
//
// Created by zhanggui on 15/8/8.
// Copyright (c) 2015年 zhanggui. All rights reserved.
// #import "ConcreteSet.h"
#import "ConcreteIterator.h"
@implementation ConcreteSet -(id)init
{
self = [super init];
if (self) {
items = [NSMutableArray new];
}
return self;
}
-(NSInteger)getCount
{
return [items count];
}
-(id)getItemFromIndex:(NSInteger)index
{
return [items objectAtIndex:index];
}
-(ConcreteIterator *)createiterator
{
return [ConcreteIterator new];
}
-(void)insertItem:(id)item
{
[items addObject:item];
}
@end

Concreteiterator.h

      	//
// ConcreteIterator.h
// IterationDemo
//
// Created by zhanggui on 15/8/8.
// Copyright (c) 2015年 zhanggui. All rights reserved.
// #import <Foundation/Foundation.h> @class ConcreteSet;
@interface ConcreteIterator : NSObject
{
ConcreteSet *mySet;
int current;
} -(id)FirstItem; -(id)nextItem; -(BOOL)isFinish; -(id)currentItem;
-(id)initWithConcreteSet:(ConcreteSet *)concrete;
@end

Concreteiterator.m

//
// ConcreteIterator.m
// IterationDemo
//
// Created by zhanggui on 15/8/8.
// Copyright (c) 2015年 zhanggui. All rights reserved.
// #import "ConcreteIterator.h"
#import "ConcreteSet.h" @implementation ConcreteIterator -(id)initWithConcreteSet:(ConcreteSet *)concrete
{
self = [super init];
if (self) {
mySet = concrete;
} return self;
} -(id)FirstItem
{
return [mySet getItemFromIndex:0];
}
-(id)nextItem
{
current ++;
if (current<[mySet getCount]) {
return [mySet getItemFromIndex:current];
}else
{
return nil;
}
}
-(BOOL)isFinish
{
return current>=[mySet getCount] ? YES : NO;
}
-(id)currentItem
{
return [mySet getItemFromIndex:current];
}
@end

然后在ViewController中测试

//
// ViewController.m
// IterationDemo
//
// Created by zhanggui on 15/8/8.
// Copyright (c) 2015年 zhanggui. All rights reserved.
// #import "ViewController.h"
#import "ConcreteIterator.h"
#import "ConcreteSet.h"
@interface ViewController () @end @implementation ViewController - (void)viewDidLoad {
[super viewDidLoad];
// NSArray *arr = @[@"apple",@"banana",@"orange"];
//
// for(NSString *item in arr)
// {
// NSLog(@"%@",item);
// }
ConcreteSet *set = [[ConcreteSet alloc] init];
[set insertItem:@"apple"];
[set insertItem:@"banana"];
[set insertItem:@"orange"]; NSLog(@"Count is :%ld",(long)[set getCount]); ConcreteIterator *iterator = [[ConcreteIterator alloc] initWithConcreteSet:set]; while (![iterator isFinish]) {
NSLog(@"%@",[iterator currentItem]);
[iterator nextItem];
}
} @end

用自己定义的迭代器去输出元素和用苹果枚举效果类似。苹果推荐使用类似与viewDidLoad方法中的for循环来输出集合元素。

附:

iOS设计模式之迭代器模式的更多相关文章

  1. iOS 设计模式之工厂模式

    iOS 设计模式之工厂模式 分类: 设计模式2014-02-10 18:05 11020人阅读 评论(2) 收藏 举报 ios设计模式 工厂模式我的理解是:他就是为了创建对象的 创建对象的时候,我们一 ...

  2. 乐在其中设计模式(C#) - 迭代器模式(Iterator Pattern)

    原文:乐在其中设计模式(C#) - 迭代器模式(Iterator Pattern) [索引页][源码下载] 乐在其中设计模式(C#) - 迭代器模式(Iterator Pattern) 作者:weba ...

  3. Python进阶:设计模式之迭代器模式

    在软件开发领域中,人们经常会用到这一个概念——“设计模式”(design pattern),它是一种针对软件设计的共性问题而提出的解决方案.在一本圣经级的书籍<设计模式:可复用面向对象软件的基础 ...

  4. 设计模式学习--迭代器模式(Iterator Pattern)和组合模式(Composite Pattern)

    设计模式学习--迭代器模式(Iterator Pattern) 概述 ——————————————————————————————————————————————————— 迭代器模式提供一种方法顺序 ...

  5. js设计模式——4.迭代器模式

    js设计模式——4.迭代器模式 代码演示 /*js设计模式——迭代器模式*/ class Iterator { constructor(container) { this.list = contain ...

  6. 实践GoF的设计模式:迭代器模式

    摘要:迭代器模式主要用在访问对象集合的场景,能够向客户端隐藏集合的实现细节. 本文分享自华为云社区<[Go实现]实践GoF的23种设计模式:迭代器模式>,作者:元闰子. 简介 有时会遇到这 ...

  7. 【GOF23设计模式】迭代器模式

    来源:http://www.bjsxt.com/ 一.[GOF23设计模式]_迭代器模式.JDK内置迭代器.内部类迭代器 package com.test.iterator; /** * 自定义的迭代 ...

  8. [设计模式] 16 迭代器模式 Iterator Pattern

    在GOF的<设计模式:可复用面向对象软件的基础>一书中对迭代器模式是这样说的:提供一种方法顺序访问一个聚合对象中各个元素,而又不需要暴露该对象的内部表示. 类图和实例: 迭代器模式由以下角 ...

  9. php设计模式之迭代器模式

    今天的PHP设计模式系列的主角是迭代器(Iterator)模式,迭代器模式提供了抽象:位于对象图不明部分的一组对象(或标量)集合上的迭代. 迭代器(Iterator)模式,它在一个很常见的过程上提供了 ...

随机推荐

  1. JavaScript备忘录(2)——闭包

    语句 JavaScript是解释型语言,解释器是按照顺序逐句执行的(除了进行一些少量预处理,如将函数声明提前). 顺序是由流程控制语句来控制的,常用的流程控制语句包括: 条件控制语句:if...els ...

  2. winform用户控件

    用途用户控件包含Time控件和一个lable控件,一个ToolStrip控件,每隔一秒显示一次时间     1. 生成用户控件   新建一个项目类型为用户控件   注意定义类名,此类名为以后工具箱中显 ...

  3. Flex Array 与 ArrayCollection.转自网络

    1.array作为控件使用 FLEX3写法: <mx:Array id="barname"> <mx:String>Flash</mx:String& ...

  4. Android 简单计算器源码....

    PS:今天算是闲着没事做了一个小型的计算器...顺便熟悉一下Android的布局,组件,以及时间监听的方法...就当是做了一个小小的练习吧...     顺便去对比了一下别人写的代码...有的使用到了 ...

  5. 利用百度词典API和Volley网络库开发的android词典应用

     关于百度词典API的说明,地址在这里:百度词典API介绍 关于android网络库Volley的介绍说明,地址在这里:Android网络通信库Volley 首先我们看下大体的界面布局!

  6. 关于eclipse中MAVEN WEB工程中编译问题

    这几天是被java的环境搞疯了,我先是搭了一个spring+springmvc+mybatis的工程,在家里跑了一下,没有问题,把工程带到公司里用,却一直不能使用. 按常理来说,只要工程发生一点变化, ...

  7. Mysql查询大表出现的一个错误

    第一次测试执行下面的语句时发生如下错误:bigdata里面有100万条数据 mysql> select count(*) from (select distinct(id) from bigda ...

  8. JdbcTemplate使用总结

    Spring JdbcTemplate 在数据库的操作中,每个业务方法都要得到连接,开启事务,提交事务,回滚,关闭连接等,我们可以把这些做成一个模版,这样,在业务代码中只需要关注业务逻辑即可. MyJ ...

  9. CS0016: 未能写入输出文件“c:\Windows\Microsoft.NET\Framework64\v4.0.30319\Temporary ASP.NET Files\helloiis\ceb8cab3\4db603d8\App_global.asax.gr73hi-k.dll”--“拒绝访问。 ”

    我的报错页面: 我是使用的第一种方法解决的. 转至http://blog.csdn.net/zyzlywq/article/details/17916799 解决方法: 1,通常的解决方法:原因是由于 ...

  10. 简单使用Dos命令关闭计算机

    作为一个刚进It行业的新手,我比较喜欢搜寻一些对我有帮助的东西,下面是用本机的dos命令关机的例子: 用window+r打开运行窗口,输入cmd,进入控制台,查找关机的相关命令,输入shutdown. ...