为什么`for...of` 循环无法输出对象的自定义属性?

for...of 循环无法输出对象的自定义属性(如 hello),是由其底层设计原理决定的。


一、核心原因:for...of 仅遍历可迭代对象的值

for...of 是 ES6 引入的语法,专门用于遍历实现了迭代器接口(Symbol.iterator )的对象(如数组、Map、Set 等)。它的工作逻辑是:

  1. 调用对象的迭代器方法const iterator = obj[Symbol.iterator]();
  2. 依次获取值iterator.next().value
  3. 不涉及对象属性的遍历

普通对象(如 {}默认没有实现迭代器接口 ,因此无法直接被 for...of 遍历。尝试遍历会直接报错:

arduino 复制代码
const obj = { hello: "world" };
for (const item of obj) {} // TypeError: obj is not iterable 

底层原理for...of 依赖对象的 @@iterator 方法(通过 Symbol.iterator 访问),普通对象原型链上无此方法 。


二、对比 for...in:为何能输出自定义属性?

for...in 的设计目标是遍历对象的可枚举属性(包括自定义属性):

  1. 遍历范围:对象自身属性 + 原型链上的可枚举属性。
  2. 输出键名:循环变量是属性名(字符串类型)。
vbnet 复制代码
const obj = { hello: "world" };
for (const key in obj) {
  console.log(key);  // 输出 "hello"
}

关键区别for...in 是属性枚举机制,而 for...of 是迭代器协议的值消费机制


三、如何让 for...of 遍历对象属性?

手动实现迭代器接口,将对象转换为可迭代结构:

javascript 复制代码
javascript
复制
const obj = {
  hello: "world",
  [Symbol.iterator]() { // 实现迭代器 
    const keys = Object.keys(this); 
    let index = 0;
    return {
      next: () => ({
        value: this[keys[index++]], // 返回属性值 
        done: index > keys.length  
      })
    };
  }
};
 
for (const value of obj) {
  console.log(value);  // 输出 "world"
}

此方法将对象的属性值转为可迭代序列 。


四、设计哲学:为何如此区分?

  1. 语义分离

    • for...in → 遍历属性名(适用于对象配置、键值对操作)
    • for...of → 遍历元素值(适用于数组、集合等数据序列)。
  2. 避免意外行为

    • for...of 不遍历原型链属性,防止污染数据。
  3. 性能优化

    • 迭代器协议(如数组)比属性枚举更高效。

总结:关键差异速查表

特性 for...of for...in
遍历目标 可迭代对象的元素值 对象的可枚举属性名
支持普通对象 ❌(需手动实现迭代器)
输出内容 元素值(如数组项、Map值) 属性名(字符串)
遍历原型链属性 ✅(需用 hasOwnProperty 过滤)

最佳实践

  • 遍历数组/Map/Set → 用 for...of(直接获取值)
  • 遍历对象属性 → 用 for...in(或 Object.keys()+for...of )。
相关推荐
killerbasd1 小时前
牧苏苏传 我不装了 4/7
前端·javascript·vue.js
橘子编程2 小时前
JavaScript与TypeScript终极指南
javascript·ubuntu·typescript
叫我一声阿雷吧2 小时前
JS 入门通关手册(45):浏览器渲染原理与重绘重排(性能优化核心,面试必考
javascript·前端面试·前端性能优化·浏览器渲染·浏览器渲染原理,重排重绘·reflow·repaint
大家的林语冰3 小时前
《前端周刊》尤大开源 Vite+ 全家桶,前端工业革命启动;尤大爆料 Void 云服务新产品,Vite 进军全栈开发;ECMA 源码映射规范......
前端·javascript·vue.js
jiayong233 小时前
第 8 课:开始引入组合式函数
前端·javascript·学习
天若有情6734 小时前
【C++原创开源】formort.h:一行头文件,实现比JS模板字符串更爽的链式拼接+响应式变量
开发语言·javascript·c++·git·github·开源项目·模版字符串
yuki_uix4 小时前
重排、重绘与合成——浏览器渲染性能的底层逻辑
前端·javascript·面试
止观止5 小时前
拥抱 ESNext:从 TC39 提案到生产环境中的现代 JS
开发语言·javascript·ecmascript·esnext
时寒的笔记5 小时前
js逆向7_案例惠nong网
android·开发语言·javascript
吴声子夜歌5 小时前
ES6——Generator函数详解
前端·javascript·es6