欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  移动技术

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

程序员文章站 2023-11-20 16:57:40
基本理解 迭代器模式(iterrator):提供一个方法顺序访问一个聚合对象中的各个元素,而又不暴露该元素的内部表示。 当你访问一个聚合对象,而且不管这些对象是什么都需...

基本理解
迭代器模式(iterrator):提供一个方法顺序访问一个聚合对象中的各个元素,而又不暴露该元素的内部表示。
当你访问一个聚合对象,而且不管这些对象是什么都需要遍历的时候,你就应该考虑用迭代器模式。
你需要对聚集有多种方式遍历时,可以考虑用迭代器模式。
迭代器模式就是分离了集合对象的遍历行为,抽象出一个迭代器类来负责,这样既可以做到不暴露集合的内部结构,又可让外部代码透明地访问集合内部的数据。
迭代器定义了一个用于访问集合元素并记录当前元素的接口。
不同的迭代器可以执行不同的迭代策略。
外部迭代器和内部迭代器:

外部迭代器

  • 外部迭代器让客户端直接操作迭代过程,所以客户端需要知道外部迭代器才能使用。但是它为客户端提供了更多的控制
  • 客户端创建并维护外部迭代器
  • 客户端可以使用不同外部迭代器实现多种类型的遍历

内部迭代器

  • 客户端不需要知道任何外部迭代器,而是可以通过集合对象的特殊接口,或者一次访问一个元素,或者向集合中的每个元素发送消息。
  • 集合对象本身创建并维护它的外部迭代器
  • 集合对象可以在不修改客户端代码的情况下,选择不同的外部迭代器

在cocoa touch框架中使用迭代器模式?

    基础框架中的nsenumerator类实现了迭代器模式。抽象nsenumerator类的私有具体子类返回枚举器对象,能够顺序遍历各种集合——数组、集合、字典,把集合中的对象返回给客户端。

    nsdirectoryenumerator,这个类的实例递归枚举文件系统中一个目录的内容。nsarray、nsset、nsdictionary这样的集合类,定义了返回与集合类型相应的nsenumerator子类实例的方法。所有的枚举器都以同样的方式工作,可以在一个循环中向枚举器发送nextobject消息,从枚举器取得对象,直到它返回nil表示遍历结束。
1.nsenumerator

    我们可以使用nsenumerator来枚举nsarray、nsdictionary和nsset对象中的元素。nsenumerator本身是个抽象类,它有依靠几个工厂方法,如objectenumrator或keyenumerator,来创建并返回相应的具体枚举器对象。代码如下:

复制代码 代码如下:

 nsarray *array = @[@"张三", @"李四", @"王五"];
    nsenumerator *itemenumerator = [array objectenumerator];
    
    nsstring *item;
    while (item = [itemenumerator nextobject]) {
        nslog(@"item is :%@", item);
    }

2015-08-28 16:48:05.463 nsenumatrodemo[55301:3712762] item is :张三
2015-08-28 16:48:05.463 nsenumatrodemo[55301:3712762] item is :李四
2015-08-28 16:48:05.464 nsenumatrodemo[55301:3712762] item is :王五

    使用nsenumerator对数组进行遍历,当消息调用[itemenumerator nextobject]会返回nil,然后枚举过程就结束了。

2.基于块的枚举

    从ios4.0后,在nsarray、nsdictionary和nsset对象中引入了新方法,用于基于块的枚举。其中一个方法叫enumerateobjectsusingblock:(void(^)(id obj, nsuinteger idx, bool *stop))block。我们可以把自己的算法定义在内嵌到消息调用之中的块里,或者在别的什么地方预定义一个块,然后作为参数传给消息调用。如下代码:

复制代码 代码如下:

nsarray *array = @[@"张三", @"李四", @"王五"];
nsstring *str = @"李四";
[array enumerateobjectsusingblock:^(id obj, nsuinteger idx, bool *stop) {
     nslog(@"item is :%@", obj);
        
     if ([obj localizedstandardcompare:str] == nsorderedsame) {
         *stop = yes;
         nslog(@"停止遍历");
     }
}];

2015-08-28 17:10:03.556 nsenumatrodemo[55478:3723216] item is :张三
2015-08-28 17:10:03.557 nsenumatrodemo[55478:3723216] item is :李四
2015-08-28 17:10:03.557 nsenumatrodemo[55478:3723216] 停止遍历

    如果array数组中有字符串"李四",那么久把指针*stop设置为yes,以通知array对象提前停止遍历。

    nsset对象中基于块的枚举与nsarray中的非常类似,只是在块的参数中没有idx参数。因为集合中的元素是无序的。

    使用nsarray、nsdictionary和nsset的内部迭代器的一个重要好处是,处理其内容的算法可以在其他地方由其他开发人员来定义。与传统的for循环中定义的算法不同,定义清晰的块可以被复用。当块逐渐变大时,可把它们放到单独的实现文件中,不跟其他代码挤在一起。

3.快速枚举

    从ios2.0后提供了一种枚举,快速枚举,也是苹果推荐的枚举方法。它允许把集合对象的枚举直接用作for循环的一部分,无需使用其他枚举对象,而且比传统的机遇索引的for循环效率更高。现在枚举循环使用指针运算,让它比使用nsenumerator的标准方法效率更高。

    要使用快速枚举,集合类需要实现nsfastenumeration协议,以向运行库提供关于集合的必要信息。基础框架中的所有集合类与nsenumerator类都支持快速枚举。因此不必使用while循环从nsenumerator枚举每个元素,直到nextobject返回nil。代码如下:

复制代码 代码如下:

nsarray *array = @[@"张三", @"李四", @"王五"];
    for (id item in array) {
        nslog(@"item is :%@", item);
    }

2015-08-28 17:28:18.619 nsenumatrodemo[55596:3730966] item is :张三
2015-08-28 17:28:18.620 nsenumatrodemo[55596:3730966] item is :李四
2015-08-28 17:28:18.620 nsenumatrodemo[55596:3730966] item is :王五

4.内部枚举

    nsarray有个实例方法叫(void)makeobjectsperformselector:(sel)aselector,它允许客户端向数组中每个元素发送一个消息,让每个元素执行指定的aselector。可以用前面提到的任何一种枚举方法让每个元素执行相同的选择器,达到相同的目的。这个方法在内部枚举集合并向每个元素发送performselector:消息。这种方式的缺点是如果集合中任何元素不响应选择器,就会抛出异常。因此它主要使用于不需要太多运行时检查的简单操作。