接着上文[js高手之路] es6系列教程 - 迭代器与生成器详解继续.

在es6中引入了一个新的循环结构for ....of, 主要是用来循环可迭代的对象,那么什么是可迭代的对象呢?

可迭代的对象一般都有Symbol.iterator属性,你可以在控制台中用console.dir打印数组,Map,Set,在他们的原型对象(prototype)上面就能找到。这个属性与迭代器密切相关,通过该函数可以返回一个迭代器,下文,我会举一个例子。一般来说所有的集合对象(数组,Set,Map 以及字符串)都是可迭代的对象。这些对象中都有默认的迭代器.

for..of循环的遍历原理:

循环每执行一次都会调用可迭代对象的next()方法, 并将迭代器返回的结果对象的value属性存储在一个变量中,循环将持续执行这一过程,直到done返回true

 let values = [ 10, 20, 30 ];
for( let val of values ) {
console.log( val );
}

这段for...of循环的代码通过调用values数组的Symbol.iterator方法来获取迭代器, 这一步是由js引擎自动完成的,随后迭代器的next()方法被多次调用,返回对象的value值并存储在变量val中,依次为: 10, 20, 30,done为true的时候退出,最后 不会把undefined赋值给val.

利用Symbol.iterator访问默认的迭代器

 let userList = [ 'ghostwu', '悟空', '八戒' ];
let iterator = userList[Symbol.iterator]();
console.log( iterator.next() ); //{ value : 'ghostwu', done : false }
console.log( iterator.next() ); //{ value : '悟空', done : false }
console.log( iterator.next() ); //{ value : '八戒', done : false }
console.log( iterator.next() ); //{ value : undefined, done : false }

由于具有Symbol.iterator属性的对象都有默认的迭代器,这个特性可以用来检测对象是否为可迭代对象.

 function isIterable( obj ){
return typeof obj[Symbol.iterator] === 'function';
}
console.log( isIterable( [ 10, 20 ,30 ] ) ); //true
console.log( isIterable( "ghostwu" ) ); //true
console.log( isIterable( new Map() ) ); //true
console.log( isIterable( new Set() ) ); //true
console.log( isIterable( new Object() ) ); //false
console.log( isIterable( {} ) );//false

所以,for..of不能遍历对象( json )

var obj = {
name : 'ghostwu',
age : 22,
sex : 'man'
};
for( var val of obj ) {
console.log( val );
}

上面这种遍历方式,就会报错.

但是,我们可以为对象增加一个迭代器方法

 let obj = {
items : [],
*[Symbol.iterator](){
for( let item of this.items ) {
yield item;
}
}
}
obj.items.push( 10 );
obj.items.push( 20 );
obj.items.push( 30 );
var iterator = obj[Symbol.iterator]();
console.log( iterator.next() ); //{ value : 10, done: false }
console.log( iterator.next() ); //{ value : 20, done : false }
console.log( iterator.next() ); //{ value : 30, done : false }
console.log( iterator.next() ); //{ value : undefined, done : true }

给Symbol.iterator属性添加一个生成器,那么对象就具备迭代器的功能了,那么就能够使用for...of方法了

 let obj = {
items : [],
*[Symbol.iterator](){
for( let item of this.items ) {
yield item;
}
}
}
obj.items.push( 10 );
obj.items.push( 20 );
obj.items.push( 30 ); for ( let val of obj ) {
console.log( val );
}

内置迭代器:

可迭代的对象,都内置以下3种迭代器

entries(): 返回一个迭代器,值为键值对

values(): 返回一个迭代器, 值为集合的值

keys(): 返回一个迭代器,值为集合中的所有键

 let userList = [ 'ghostwu', '悟空', '八戒' ];

 for ( let name of userList.entries() ) {
console.log( name );
} let set = new Set( [ 10, 20, 30 ] );
for ( let num of set.entries() ){
console.log( num );
} let map = new Map( [ [ 'name', 'ghostwu' ], [ 'age', 22 ] ] );
for ( let detail of map.entries() ){
console.log( detail );
}

entries返回的是键值对,注意Set集合,返回的键和值是一样的.

 let set = new Set( [ 10, 20, 30 ] );
for ( let num of set.values() ){
console.log( num );
} let map = new Map( [ [ 'name', 'ghostwu' ], [ 'age', 22 ] ] );
for ( let detail of map.values() ){
console.log( detail );
}

 let set = new Set( [ 10, 20, 30 ] );
for ( let num of set.keys() ){
console.log( num );
} let map = new Map( [ [ 'name', 'ghostwu' ], [ 'age', 22 ] ] );
for ( let detail of map.keys() ){
console.log( detail );
}

 默认的迭代器:

 let userList = [ 'ghostwu', '悟空', '八戒' ];

 //相当于调用values
for ( let name of userList ) {
console.log( name );
} let set = new Set( [ 10, 20, 30 ] );
//相当于调用values
for ( let num of set ){
console.log( num );
} let map = new Map( [ [ 'name', 'ghostwu' ], [ 'age', 22 ] ] );
//相当于调用entries
for ( let detail of map ){
console.log( detail );
}

Map的默认行为,可以用解构来简写:

 let map = new Map( [ [ 'name', 'ghostwu' ], [ 'age', 22 ] ] );
for ( let [ key, value ] of map ) {
console.log( key + '--->' + value );
}

利用展开运算符把Set与Map转化成数组

 let set = new Set( [ 10, 20, 30 ] );
let arr = [...set];
console.log( arr ); // [10,20,30] let map = new Map( [ [ 'name', 'ghostwu' ], [ 'age', 22 ] ]);
console.log( [...map] ); // [ [ 'name', 'ghostwu' ], [ 'age', 22 ] ] let arr1 = [ 10, 20, 30 ];
let arr2 = [ 'ghostwu', '八戒', '悟空' ];
let combine = [ ...arr1, ...arr2, 'done' ];
console.log( combine ); // [10, 20, 30, "ghostwu", "八戒", "悟空", "done"]

[js高手之路] es6系列教程 - 迭代器,生成器,for...of,entries,values,keys等详解的更多相关文章

  1. [js高手之路] es6系列教程 - 迭代器与生成器详解

    什么是迭代器? 迭代器是一种特殊对象,这种对象具有以下特点: 1,所有对象都有一个next方法 2,每次调用next方法,都会返回一个对象,该对象包含两个属性,一个是value, 表示下一个将要返回的 ...

  2. [js高手之路] es6系列教程 - 对象功能扩展详解

    第一:字面量对象的方法,支持缩写形式 //es6之前,这么写 var User = { name : 'ghostwu', showName : function(){ return this.nam ...

  3. [js高手之路] es6系列教程 - promise常见用法详解(resolve,reject,catch,then,all,race)

    关于promise我在之前的文章已经应用过好几次,如[js高手之路]Node.js+jade+express+mongodb+mongoose+promise实现todolist,本文就来讲解下pro ...

  4. [js高手之路] es6系列教程 - 函数的默认参数详解

    在ES6之前,我们一般用短路表达式处理默认参数 function show( a, b ){ var a = a || 10; var b = b || 20; console.log( a, b ) ...

  5. [js高手之路] es6系列教程 - 箭头函数详解

    箭头函数是es6新增的非常有意思的特性,初次写起来,可能会觉得别扭,习惯之后,会发现很精简. 什么是箭头函数? 箭头函数是一种使用箭头( => )定义函数的新语法, 主要有以下特性: 不能通过n ...

  6. [js高手之路] es6系列教程 - var, let, const详解

    function show( flag ){ console.log( a ); if( flag ){ var a = 'ghostwu'; return a; } else { console.l ...

  7. [js高手之路] es6系列教程 - 不定参数与展开运算符(...)

    三个点(...)在es6中,有两个含义: 用在形参中, 表示传递给他的参数集合, 类似于arguments, 叫不定参数. 语法格式:  在形参面前加三个点( ... ) 用在数组前面,可以把数组的值 ...

  8. [js高手之路]es6系列教程 - 解构详解

    解构通俗点说,就是通过一种特定格式,快捷的读取对象/数组中的数据的方法, es6之前,我们通过对象名称[键] 读取数据 var User = { 'name' : 'ghostwu', 'age' : ...

  9. [js高手之路] es6系列教程 - Set详解与抽奖程序应用实战

    我们还是从一些现有的需求和问题出发,为什么会有set,他的存在是为了解决什么问题? 我们看一个这样的例子,为一个对象添加键值对 var obj = Object.create( null ); obj ...

随机推荐

  1. 二叉树的递归遍历 Tree UVa548

    题意:给一棵点带权的二叉树的中序和后序遍历,找一个叶子使得他到根的路径上的权值的和最小,如果多解,那该叶子本身的权值应该最小 解题思路:1.用getline()输入整行字符,然后用stringstre ...

  2. jenkins+github持续集成中的坑

    1.前言 刚开始开发自己的独立博客的时候,每次发布都要手动打包,上传服务器,杀tomcat进程,重启,来回这么重复性工作,很快就有点不耐烦了.如果能自动化的东西,就绝不要手动了,所以自己搭建了个持续集 ...

  3. iOS_应用程序的生命周期

    每个iPhone程序都包括唯一一个UIApplication对象,它管理整个程序的生命周期,从载入第一个显示界面開始,而且监听系统事件.程序事件调度整个程序的运行. int main(int argc ...

  4. 哈希表(散列)HashTable实现

    近期刷Leetcode发现凡是找字符串中反复字符或者数组中找反复数据的时候就不知道从何下手了. 所以决定学习一下哈希表解题.哈希表的原理主要是解决分类问题,hash表是介于链表和二叉树之间的一种中间结 ...

  5. nyist oj 756 重建二叉树

    重建二叉树 时间限制:1000 ms  |  内存限制:65535 KB 难度:3 描写叙述 题目非常easy.给你一棵二叉树的后序和中序序列,求出它的前序序列(So easy!). 输入 输入有多组 ...

  6. hdu 4883 区间选点

    昨天比赛的时候没有做出来,本来是想用贪心的,可是贪了好久都没有招, 今天在网上搜了解题报告~好像说这是一类区间选点问题: 有一个好的做法: (1)首先把题目中的时间全转化为分钟,那么区间就在0-144 ...

  7. hdu5303Delicious Apples

    题意大概就是有n框苹果放在长度为L的环上,每框有ai个苹果.你有一个容量为k的框.要你从0点处出发,随意走.框满了就回到0点把苹果放在那里.继续走直到把苹果都拿完为止.问你最少要走多少路程. 首先贪心 ...

  8. 初识ajax

    ajax优势:不刷新整个页面,只刷新局部(无刷新) 无刷新的好处: 只更新部分页面,有效利用宽带 提供连续的用户体验 提供类似C/S的交互效果,操作更方面 什么是ajax AJAX :代表 Async ...

  9. 前端优化之动画为什么要尽量用css3代替js

    导致JavaScript效率低的两大原因:操作DOM和使用页面动画.通常我们会通过频繁的操作 DOM的CSS来实现视觉上的动画效果,导致js效率低的两个因素都包括在内了在频繁的操作DOM和CSS时,浏 ...

  10. Java中进制的转换函数

    十进制转成十六进制: Integer.toHexString(int i) 十进制转成八进制 Integer.toOctalString(int i) 十进制转成二进制 Integer.toBinar ...