迭代器模式是指提供一种方法顺序访问一个聚合对象中的各个元素,而又不需要暴露该对象的内部表示。
迭代器模式可以把迭代的过程从业务逻辑中分离出来,在使用迭代器模式之后,即使不关心对象的内部构造,也可以按顺序访问其中的每个元素
许多浏览器都支持 Javascript 的 Array.prototype.forEach
迭代器可以分为 内部迭代器 和 外部迭代器

一、jQuery 中的迭代器
 $.each( [1,2,3,4], function (i, n) {
console.log( "当前下表为:" + i + " , 当前值为:" + n );
});
二、自实现一个迭代器
 // 自己实现一个数组迭代器
var each = function( arry, callback ){
for( var i = 0, l = arry.length; i < l; i++){
callback.call( arry[i], i, arry[ i ]);
}
};
each([1,2,3,4], function (i, n) {
console.log( i + " - " + n ); // 输出数组下标和值
});
三、内部迭代器

  上边的 each 函数属于内部迭代器, each 函数内部已经定义好了迭代原则,它完全接手整个迭代过程,外部只需要一次初始调用。
  内部迭代器的优点也刚好是它的缺点 - 使用方便,迭代交互也仅仅是一次初始调用
  示例: 在不改写 each 本身的代码前提,实现判断两个数组里元素的值是否完全相等
 //判断两个数组的值是否完全相等
var compare = function( arry1, arry2){
if( arry1.length !== arry2.length ){
throw new Error( "arry1和arry2不相等" );
}
each( arry1, function( i, n ){
if( n !== arry2[i] ){
throw new Error("arry1和arry2不相等");
}
});
console.log( "arry1和arry2相等" );
}; compare( [1,2,3,4], [1,2,3]);

 四、外部迭代器
  外部迭代器必须显式地请求迭代下一个元素
  外部迭代器增加了一些调用的复杂度,但相对的也增强了迭代器的灵活性,我们可以手工控制迭代的过程或者顺序
  示例: 重写 compare     外部迭代器:
 // 外部迭代器
var Iterator = function (obj) {
var current = 0; var next = function(){
current += 1;
}; var isDone = function(){
return current >= obj.length;
}; var getCurrItem = function(){
return obj[ current ];
}; return {
next: next,
isDone: isDone,
getCurrItem: getCurrItem
}
}


    改写 Compare   
 // 改写 Compare
var compare = function( iterator1, iterator2 ){
while( iterator1.isDone() && iterator2.isDone() ){
if( iterator1.getCurrItem() !== iterator2.getCurrItem() ){
throw new Error( "iterator1 和 iterator2不相等" );
}
iterator1.next();
iterator2.next();
}
console.log( "iterator1 和 iterator2相等" );
}; var iterator1 = Iterator( [1,2,3,4] );
var iterator2 = Iterator( [1,2,3,4] ); compare(iterator1,iterator2);

 五、中止迭代器
  重写 each 函数实现中止迭代
 // 重写 each 函数实现中止迭代
var each = function( arry, callback ){
for( var i = 0, l = arry.length; i < l; i++ ){
// callback 的执行结果返回false,提前中止迭代
if( callback( i, arry[i] ) === false ){
break;
}
}
};
each( [1,2,3,4,5], function ( i, n ) {
if( n>3 ){ // n 大于3的时候中止循环
return false;
}
console.log(n); // 输出 1 2 3
});

 六、迭代器应用示例
  目的:根据不同的浏览器获取相应的上传组件对象
    
    将不同的上传对象封装到各自的函数里; 如果函数可用,则返回该对象,否则返回false,提示迭代器继续
 // 将不同的上传对象封装到各自的函数里; 如果函数可用,则返回该对象,否则返回false,提示迭代器继续
var getActiveUploadObj = function(){
try{
return new ActiceXObject( "TXFTNActiveX.FTNUpload" ); // IE 上传控件
}catch(e){
return false;
}
};
var getFlashUploadObj = function(){
if( supportFlash() ){
var str = "<object type='application/x-shockwave-flash'></object>";
return $( str).appendTo( $("body") );
}
return false;
};
var getFormUpl0adObj = function(){
var str = "<input type='file' type='file' class='ui-file' />"; // 表单上传
return $( str).appendTo( $("body") );
};
     //迭代器代码
 //迭代器代码
var iteratorUploadObj = function(){
for( var i = 0, fn; fn = arguments[ i++ ]; ){
var uploadObj = fn();
if( uploadObj !== false ){
return uploadObj;
}
}
}; var uploadObj = iteratorUploadObj( getActiveUploadObj, getFlashUploadObj, getFormUpl0adObj );

七、总结
迭代模式相对简单,简单到很多时候我们不认为它是一种设计模式
阅读参考书籍 - << JavaScript 设计模式与开发实践 >>

最新文章

  1. LeetCode-Combinations
  2. iOS 开发多线程篇—GCD的常见用法
  3. DSO、CUBE区别(覆盖、合计)
  4. Mongodb For C# &quot;Query&quot; 对象常用的方法
  5. [cdoj843] 冰雪奇缘 (线段树+离散)
  6. Linux下tail命令
  7. Oracle SQL Developer 操作
  8. [置顶] 与小伙伴共勉的java有关jvm的知识(一),小鸟尽量写得详细哦,欢迎讨论,谢绝喷子
  9. UEditor使用------图片上传与springMVC集成 完整实例
  10. Windows Redis默认配置文件,Redis配置不生效解决方案
  11. SSM-Spring-11:Spring中使用代理工厂Bean实现aop的四种增强
  12. python 试题
  13. 为什么越来越少的人用jQuery
  14. 牛客挑战赛30 小G砍树 树形dp
  15. 转载:MySQL EXPLAIN 命令详解学习
  16. 关于django的一些基础知识
  17. Netty自带连接池的使用
  18. cookie和session必须了解的东西
  19. hdu-6434-欧拉函数
  20. (转)Eclipse导入EPF配置文件

热门文章

  1. Android中的Uri.parse()
  2. phpMyadmin /scripts/setup.php Remote Code Injection &amp;&amp; Execution CVE-2009-1151
  3. Xcode的一些有用的插件
  4. APNs详细使用步骤
  5. 根据.MDF文件查看 SQL数据库的版本信息
  6. [Angularjs]国际化
  7. 分子量 (Molar Mass,ACM/ICPC Seoul 2007,UVa 1586)
  8. serialization机制
  9. input file美化
  10. Cocoa的MVC架构分析 cocoa的mvc实现