[js高手之路] es6系列教程 - 迭代器,生成器,for...of,entries,values,keys等详解
接着上文[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等详解的更多相关文章
- [js高手之路] es6系列教程 - 迭代器与生成器详解
什么是迭代器? 迭代器是一种特殊对象,这种对象具有以下特点: 1,所有对象都有一个next方法 2,每次调用next方法,都会返回一个对象,该对象包含两个属性,一个是value, 表示下一个将要返回的 ...
- [js高手之路] es6系列教程 - 对象功能扩展详解
第一:字面量对象的方法,支持缩写形式 //es6之前,这么写 var User = { name : 'ghostwu', showName : function(){ return this.nam ...
- [js高手之路] es6系列教程 - promise常见用法详解(resolve,reject,catch,then,all,race)
关于promise我在之前的文章已经应用过好几次,如[js高手之路]Node.js+jade+express+mongodb+mongoose+promise实现todolist,本文就来讲解下pro ...
- [js高手之路] es6系列教程 - 函数的默认参数详解
在ES6之前,我们一般用短路表达式处理默认参数 function show( a, b ){ var a = a || 10; var b = b || 20; console.log( a, b ) ...
- [js高手之路] es6系列教程 - 箭头函数详解
箭头函数是es6新增的非常有意思的特性,初次写起来,可能会觉得别扭,习惯之后,会发现很精简. 什么是箭头函数? 箭头函数是一种使用箭头( => )定义函数的新语法, 主要有以下特性: 不能通过n ...
- [js高手之路] es6系列教程 - var, let, const详解
function show( flag ){ console.log( a ); if( flag ){ var a = 'ghostwu'; return a; } else { console.l ...
- [js高手之路] es6系列教程 - 不定参数与展开运算符(...)
三个点(...)在es6中,有两个含义: 用在形参中, 表示传递给他的参数集合, 类似于arguments, 叫不定参数. 语法格式: 在形参面前加三个点( ... ) 用在数组前面,可以把数组的值 ...
- [js高手之路]es6系列教程 - 解构详解
解构通俗点说,就是通过一种特定格式,快捷的读取对象/数组中的数据的方法, es6之前,我们通过对象名称[键] 读取数据 var User = { 'name' : 'ghostwu', 'age' : ...
- [js高手之路] es6系列教程 - Set详解与抽奖程序应用实战
我们还是从一些现有的需求和问题出发,为什么会有set,他的存在是为了解决什么问题? 我们看一个这样的例子,为一个对象添加键值对 var obj = Object.create( null ); obj ...
随机推荐
- Codeforces Round #436 (Div. 2)
http://codeforces.com/contest/864 第一次打cf的月赛-- A 题意:给你一个数列,问你能不能保证里面只有两种数且个数相等.2<=n<=100,1<= ...
- Oracle在本地调试成功读取数据,但是把代码放到服务器读不出数据的解决方法。
用MVC EF框架开发项目,数据库用的是Oracle,本地调试的时候一切正常,但是把代码编译之后放到服务器就会读不出数据. 原因:本地调试环境与服务器环境不一致. 办法:在服务器上装ODT.NET组件 ...
- javaweb学习总结(七)——HttpServletResponse对象(一)(转)
转载自 http://www.cnblogs.com/xdp-gacl/p/3789624.html Web服务器收到客户端的http请求,会针对每一次请求,分别创建一个用于代表请求的request对 ...
- eclipse 鲜为人知的调试技巧,你用过多少
今天在OSChina上看到了篇调试技巧,的确对于调试非常有帮助,而且大部分我们都没实用过,我们常常使用的调试是F5678四个键,假设你还想提高调试效率你能够尝试着用一用,写过代码做过项目的人都知道调试 ...
- SharePoint Framework 简介
作者:陈希章 发表于 2017年12月25日 前言 通过前面几篇文章,我相信大家对于SharePoint Online的开发有了更加全面的认识,上一篇 介绍的SharePoint Add-in的开发, ...
- 【SqlServer系列】表达式(expression)
1 概述 本篇这文章主要概述SqlServer表达式. 2 具体内容 2.1 使用范围 SQL Server(2008开始) :Azure SQL数据库:Azure SQL数据仓库:并行数 ...
- WPF中展示HTML
业务需求:将具有表格信息的HTML片段在WPF中展示出来,并像网页端一样,可以进行input的填写,checkbox选择,最后以HTML的形式完成保存. 天真的以为直接引入WPF中的WebBrowse ...
- 解决 ASP.NET Core MySql varchar 字符串截取(长度 255)
ASP.NET Core 中使用 MySql,如果字段类型为varchar,不管设置多少长度,插入或更新数据的时候,会自动截断(截取 255 长度的字符). 出现问题的原因,就是使用了MySql.Da ...
- 安装虚拟机后无法SSH远程连接
1.安装虚拟机工具 vmware workstation 2.创建一个虚拟机,系统版本是:CentOS-6.8-x86_64-LiveDVD 3.系统安装完成后,选择网络为桥接模式,如图 4.检查主机 ...
- Netty学习笔记(一):接收nodejs模拟表单上传的文件
好久不写博客了,也好久不写代码了,这两天临时遇上一个事情,觉得不难,加上觉得手有些生,就动手做了一下,结果遇上了不少坑,有新坑,有老坑,痛苦无比,现在总算差不多了,赶紧记录下来,希望以后不再重复这种痛 ...