美文网首页
23- for...of 循环遍历

23- for...of 循环遍历

作者: 夏海峰 | 来源:发表于2018-08-22 10:27 被阅读13次

    ES6 借鉴 C++、Java、C# 和 Python 语言,引入了for...of循环,作为遍历所有数据结构的统一方法。

    一个数据结构只要部署了Symbol.iterator属性,就被视为具有 iterator 接口,就可以用for...of循环遍历它的成员。也就是说,for...of循环内部调用的是数据结构的Symbol.iterator方法。

    for...of循环可以使用的范围包括数组、Set 和 Map 结构、某些类似数组的对象(比如arguments对象、DOM NodeList 对象)、Generator 对象,以及字符串。

    1、for...of 遍历数组

    数组原生具备iterator接口(即默认部署了Symbol.iterator属性),for...of循环本质上就是调用这个接口产生的遍历器,可以用下面的代码证明。

    const arr ['red', 'green', 'blue'];
    for (let v of arr) {
        console.log(v);  // red  green  blue
    }
    
    const obj = {};
    obj[Symbol.iterator] = arr[Symbol.iterator].bind(arr);
    for (let o of obj) {
        console.log(o);  // red  green  blue
    }
    

    for...of循环可以代替数组实例的forEach方法。for...in循环读取键名,for...of循环读取键值。

    2、for...of 遍历 Map 和 Set 结构

    Set 和 Map 结构也原生具有 Iterator 接口,可以直接使用for...of循环。

    var engines = new Set(["Gecko", "Trident", "Webkit", "Webkit"]);
    for (var e of engines) {
      console.log(e);
    }
    // Gecko
    // Trident
    // Webkit
    
    var es6 = new Map();
    es6.set("edition", 6);
    es6.set("committee", "TC39");
    es6.set("standard", "ECMA-262");
    for (var [name, value] of es6) {
      console.log(name + ": " + value);
    }
    // edition: 6
    // committee: TC39
    // standard: ECMA-262
    

    3、for...of 遍历 数组/Map/Set 之 keys()/values()/entries() 返回的对象

    数组、Set、Map 都部署了keys()/values()/entries()三个方法,调用这三个方法后都返回遍历器对象。使用 for...of 可以遍历它们的返回结果。

    let arr = ['a', 'b', 'c'];
    for (let pair of arr.entries()) {
      console.log(pair);
    }
    // [0, 'a']
    // [1, 'b']
    // [2, 'c']
    

    4、for...of 遍历字符串

    let str = 'hello';
    
    for (let s of str) {
        console.log(s);   // h  e  l  l  o
    }
    

    5、for...of 遍历 NodeList对象

    let opArr = document.querySelectorAll('p');
    
    for(let p of opArr) {
        p.classList.add('on');
    }
    

    6、for...of 遍历 arguments对象

    function printArgs() {
        for (let x of arguments) {
            console.log(x);
        }
    }
    printArgs('a', 'b');
    // 'a'
    // 'b'
    

    7、for...of 遍历 非 Iterator 的类数组对象

    并不是所有类似数组的对象都具有 Iterator 接口,一个简便的解决方法,就是使用Array.from方法将其转为数组。

    let arrayLike = { length: 2, 0: 'a', 1: 'b' };
    
    // 报错
    for (let x of arrayLike) {
      console.log(x);
    }
    
    // 正确
    for (let x of Array.from(arrayLike)) {
      console.log(x);
    }
    

    8、for...of 遍历 对象
    对于普通的对象,for...of结构不能直接使用,会报错。使用 for...in 可以遍历对象的键名。

    let obj = {
        a: 1,
        b: 2,
        c: 3
    }
    
    for (let e in obj) {
        console.log(e);  // 'a'  'b'  'c'
    }
    

    总之,for...in循环主要是为遍历对象而设计的,不适用于遍历数组。

    一种解决方法是,使用Object.keys方法将对象的键名生成一个数组,然后再用 for...of 遍历这个数组。

    for(let key of Object.keys(obj)) {
        console.log(key + ': ' + obj[key]);
    }
    

    另一个方法是使用 Generator 函数将对象重新包装一下。

    function* entries(obj) {
        for(let key of Object.keys(obj)) {
            yield [key, obj[key]];
        }
    }
    for(let [key, value] of entries(obj)) {
        console.log(key, '->', value);
    }
    

    9、for...of 可以与 break / continue / return 配合使用

    for (var n of arr) {
        if (n > 10) {
            break;
        }
        console.log(n);
    }
    

    完!!!

    相关文章

      网友评论

          本文标题:23- for...of 循环遍历

          本文链接:https://www.haomeiwen.com/subject/veeeiftx.html