迭代器模式:指提供一种方法顺序访问一个聚合对象中的各个元素,而又不需要暴露该对象的内部表示。

迭代器模式可以把迭代的过程从业务逻辑中分离出来,在使用迭代器模式之后,即使不关心对象的内部构造,也可以按顺序访问其中的每个元素。

流行语言如Java、Ruby 等都已经有了内置的迭代器实现,许多浏览器也支持JavaScript的Array.prototype.forEach。

jQuery中的迭代器

 $.each( [1, 2, 3], function( i, n ){
  console.log( '当前下标为: '+ i );
  console.log( '当前值为:' + n );
});

实现自己的迭代器

 var each = function( ary, callback ){
  for ( var i = 0, l = ary.length; i < l; i++ ){
    callback.call( ary[i], i, ary[ i ] ); // 把下标和元素当作参数传给callback 函数
  }
}; each( [ 1, 2, 3 ], function( i, n ){
  alert ( [ i, n ] );
});

内部迭代器和外部迭代器

·内部迭代器
我们刚刚编写的each函数属于内部迭代器,each函数的内部已经定义好了迭代规则,它完全接手整个迭代过程,外部只需要一次初始调用。
内部迭代器在调用的时候非常方便,外界不用关心迭代器内部的实现,但由于内部迭代器的迭代规则已经被提前规定,上面的each函数就无法同时迭代2个数组了。

在不改变each函数情况下,如果要比较两个数组是否完全一致就得这样写

 var compare = function( ary1, ary2 ){
if ( ary1.length !== ary2.length ){
throw new Error ( 'ary1 和ary2 不相等' );
}
each( ary1, function( i, n ){
if ( n !== ary2[ i ] ){
throw new Error ( 'ary1 和ary2 不相等' );
}
});
alert ( 'ary1 和ary2 相等' );
}; compare( [ 1, 2, 3 ], [ 1, 2, 4 ] ); // throw new Error ( 'ary1 和ary2 不相等' );

·外部迭代器

外部迭代器必须显式地请求迭代下一个元素,外部迭代器增加了一些调用的复杂度,但我们也可以手工控制迭代的过程或者顺序。

 var Iterator = function( obj ){
  var current = 0; //指代当前序号
  var next = function(){ //next方法,序号+1
    current += 1;
  };
  var isDone = function(){ //isDone方法,判断是否数组遍历结束
    return current >= obj.length;
  };
  var getCurrItem = function(){ //getCurrItem方法,获得当前序号的具体值
    return obj[ current ];
  };
  return { //返回一个对象,利用闭包保存current的值
    next: next,
    isDone: isDone,
    getCurrItem: getCurrItem
  }
}; var iterator_1 = Iterator( [ 1, 2, 3 ] ); //生成实例
var iterator_2 = Iterator( [ 1, 2, 3 ] ); var compare = function( iterator1, iterator2 ){
  while( !iterator1.isDone() && !iterator2.isDone() ){ //当两者都没有遍历结束时
    if ( iterator1.getCurrItem() !== iterator2.getCurrItem() ){ //如果两者当前序号的值不等
      throw new Error ( 'iterator1 和iterator2 不相等' );
    }
    iterator1.next(); //序号增加
    iterator2.next();
  }
  alert ( iterator1+'和'+iterator2+'相等' );
} compare( iterator_1, iterator_2 ); // 输出:iterator_1 和iterator_2 相等

迭代类数组对象和字面量对象

无论是内部迭代器还是外部迭代器,只要被迭代的聚合对象拥有length 属性而且可以用下标访问,那它就可以被迭代。
JavaScript中,for in语句可以用来迭代普通字面量对象的属性。jQuery中的$.each函数如下

 $.each = function( obj, callback ) {
  var value,
    i = 0,
    length = obj.length,
    isArray = isArraylike( obj );
  if ( isArray ) { // 迭代类数组
    for ( ; i < length; i++ ) {
      value = callback.call( obj[ i ], i, obj[ i ] );
      if ( value === false ) {
        break;
      }
    }
  } else {
    for ( i in obj ) { // 迭代object 对象
      value = callback.call( obj[ i ], i, obj[ i ] );
      if ( value === false ) {
        break;
      }
    }
  }
  return obj;
};

倒序迭代器

遍历时候反着来就可以了

中止迭代器

给个条件,让迭代器终止。
jQuery的each 函数里有这样一句:

 if ( value === false ) {
break;
}

这句代码的意思是,约定如果回调函数的执行结果返回false,则提前终止循环。

总结

迭代器模式是一种相对简单的模式,简单到很多时候我们都不认为它是一种设计模式。目前的绝大部分语言都内置了迭代器。

javascript设计模式与开发实践阅读笔记(7)——迭代器模式的更多相关文章

  1. javascript设计模式与开发实践阅读笔记(4)——单例模式

    定义 单例模式:保证一个类仅有一个实例,并提供一个访问它的全局访问点. 具体来说,就是保证有些对象有且只有一个,比如线程池.全局缓存.浏览器中的window 对象等.在js中单例模式用途很广,比如登录 ...

  2. javascript设计模式与开发实践阅读笔记(8)——观察者模式

    发布-订阅模式,也叫观察者模式:定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都将得到通知. 在JavaScript开发中,我们一般用事件模型来替代传统的观察者模式. ...

  3. javascript设计模式与开发实践阅读笔记(6)——代理模式

    代理模式:是为一个对象提供一个代用品或占位符,以便控制对它的访问. 代理模式的关键是,当客户不方便直接访问一个对象或者不满足需要的时候,提供一个替身对象来控制对这个对象的访问,客户实际上访问的是替身对 ...

  4. javascript设计模式与开发实践阅读笔记(5)——策略模式

    策略模式:定义一系列的算法,把它们一个个封装起来,并且使它们可以相互替换. 我的理解就是把各种方法封装成函数,同时存在一个可以调用这些方法的公共函数.这样做的好处是可以消化掉内部的分支判断,使代码效率 ...

  5. javascript设计模式与开发实践阅读笔记(9)——命令模式

    命令模式:有时候需要向某些对象发送请求,但是并不知道请求的接收者是谁,也不知道被请求的操作是什么,此时希望用一种松耦合的方式来设计软件,使得请求发送者和请求接收者能够消除彼此之间的耦合关系. 说法很复 ...

  6. javascript设计模式与开发实践阅读笔记(11)—— 模板方法模式

    模板方法模式: 由两部分结构组成,第一部分是抽象父类,第二部分是具体的实现子类.通常在抽象父类中封装了子类的算法框架,包括实现一些公共方法以及封装子类中所有方法的执行顺序.子类通过继承这个抽象类,也继 ...

  7. JavaScript设计模式与开发实践——读书笔记1.高阶函数(上)

    说来惭愧,4个多月未更新了.4月份以后就开始忙起来了,论文.毕设.毕业旅行等七七八八的事情占据了很多时间,毕业之后开始忙碌的工作,这期间一直想写博客,但是一直没能静下心写.这段时间在看<Java ...

  8. 《JavaScript设计模式与开发实践》笔记第八章 发布-订阅模式

    第八章 发布-订阅模式 发布-订阅模式描述 发布-订阅模式又叫观察者模式,它定义对象间的一种一对多的依赖关系,当一个对象的状态发生改变时,所有依赖于它的对象都将得到通知. 发布-订阅模式可以广泛应用于 ...

  9. JavaScript设计模式与开发实践——读书笔记1.高阶函数(下)

    上部分主要介绍高阶函数的常见形式,本部分将着重介绍高阶函数的高级应用. 1.currying currying指的是函数柯里化,又称部分求值.一个currying的函数会先接受一些参数,但不立即求值, ...

随机推荐

  1. Python全栈之路---运算符与基本的数据结构

    运算符 一.算术运算符: 练习: + 加法 两个对象相加 1 + 2得到3:'a' + 'b'得到'ab'. - 减法 一个数减去另一个数或者是负数 5 - 3得到2:-2得到一个负数 * 乘法 两个 ...

  2. 决策树算法(1)含java源代码

    信息熵:变量的不确定性越大,熵越大.熵可用下面的公式描述:-(p1*logp1+p2*logp2+...+pn*logpn)pi表示事件i发生的概率ID3:GAIN(A)=INFO(D)-INFO_A ...

  3. Android学习之 Intent详解

    一. Intent 作用 Intent 是一个将要执行的动作的抽象的描述,一般来说是作为参数来使用,由Intent来协助完成android各个组件之间的通讯.比如说调用startActivity()来 ...

  4. Wpf 中使用gif格式的动态图

    第一种方法:使用winform插件 <WindowsFormsHost  xmlns:wf="clr-namespace:System.Windows.Forms;assembly=S ...

  5. hihoCoder 1305 区间求差

    #1305 : 区间求差 时间限制:10000ms 单点时限:1000ms 内存限制:256MB 描述 给定两个区间集合 A 和 B,其中集合 A 包含 N 个区间[ A1, A2 ], [ A3,  ...

  6. Daily Scrum 12.1

    今日完成任务: 完成了对源代码结构的修改,删除冗余等:和其他小组讨论了关于整合的问题,向其他小组介绍自己小组使用的数据库等. 明日任务: 晏旭瑞 初步完成文档上传下载 孙思权 深入了解数据库中每个表, ...

  7. 基于PHP使用rabbitmq实现消息队列

    1.从github上面获取AMQP基于php的实现扩展 2.创建生产者 send.php   3.创建消费者 receive.php 4.在cli模式下 分别执行 send.php  receive. ...

  8. alert()、confirm()和prompt()的区别

    1.警告消息框alertalert 方法有一个参数,即希望对用户显示的文本字符串.该字符串不是 HTML 格式.该消息框提供了一个“确定”按钮让用户关闭该消息框,并且该消息框是模式对话框,也就是说,用 ...

  9. Android的项目不能直接引用可移植类库的项目解决方法

    深圳_exception() 10:25:51 Android的项目不能直接引用可移植类库的项目,但是可以引用可移植类库生成的dll,这就意味着无法直接断电调试可移植类库上海-黄药师() 10:26: ...

  10. js页面刷新、前进、后退

    1.使用按钮 <input type=button value=刷新 onclick="window.location.reload()"> <input typ ...