JavaScript 中的 for...in 和 for...of 循环详解
在 JavaScript 中,for...in
和 for...of
是两种常用的循环结构,但它们有着不同的用途和行为。很多初学者容易混淆这两者,本文将详细解析它们的区别、适用场景以及注意事项。
目录
-
for…in 循环
- 基本用法
- 遍历对象属性
- 注意事项
-
for…of 循环
- 基本用法
- 可迭代对象
- 与数组的结合使用
-
主要区别对比
-
何时使用哪种循环
-
实际应用示例
-
总结
for…in 循环
基本用法
for...in
语句用于遍历对象的可枚举属性(包括原型链上的可枚举属性)。
for (variable in object) {// 执行的代码
}
遍历对象属性
const person = {name: 'Alice',age: 25,occupation: 'Developer'
};for (const key in person) {console.log(`${key}: ${person[key]}`);
}
// 输出:
// name: Alice
// age: 25
// occupation: Developer
注意事项
-
遍历顺序不保证:
for...in
不保证属性的遍历顺序,特别是在不同 JavaScript 引擎中可能有差异。 -
会遍历原型链上的属性:
function Person() {this.name = 'Bob'; } Person.prototype.age = 30;const bob = new Person();for (const prop in bob) {console.log(prop); // 输出 name 和 age }
可以使用
hasOwnProperty()
来过滤:for (const prop in bob) {if (bob.hasOwnProperty(prop)) {console.log(prop); // 只输出 name} }
-
不推荐用于数组:
const arr = ['a', 'b', 'c'];for (const index in arr) {console.log(index); // 输出 "0", "1", "2" }
虽然可以工作,但有潜在问题(如可能遍历到非数字属性)。
for…of 循环
基本用法
for...of
语句在可迭代对象(包括 Array, Map, Set, String, TypedArray, arguments 对象等)上创建一个迭代循环。
for (variable of iterable) {// 执行的代码
}
可迭代对象
for...of
只能用于实现了 [Symbol.iterator]
方法的对象:
const arr = ['a', 'b', 'c'];for (const value of arr) {console.log(value); // 输出 "a", "b", "c"
}const str = 'hello';
for (const char of str) {console.log(char); // 输出 "h", "e", "l", "l", "o"
}const set = new Set([1, 2, 3]);
for (const num of set) {console.log(num); // 输出 1, 2, 3
}const map = new Map([['a', 1], ['b', 2]]);
for (const [key, value] of map) {console.log(`${key}: ${value}`); // 输出 "a: 1", "b: 2"
}
与数组的结合使用
for...of
是遍历数组元素的理想选择:
const fruits = ['apple', 'banana', 'orange'];for (const fruit of fruits) {console.log(fruit);
}
// 输出:
// apple
// banana
// orange
主要区别对比
特性 | for…in | for…of |
---|---|---|
用途 | 遍历对象属性 | 遍历可迭代对象的值 |
返回值 | 属性名(key) | 属性值(value) |
可遍历对象 | 任意对象 | 可迭代对象(Array, Map等) |
遍历原型链属性 | 是 | 否 |
数组遍历 | 遍历索引(字符串) | 遍历元素值 |
性能 | 相对较慢 | 相对较快 |
引入版本 | ES1 | ES6 |
何时使用哪种循环
-
使用
for...in
当:- 需要遍历对象的所有可枚举属性
- 需要检查对象是否具有某些属性
- 需要遍历原型链上的属性(或明确使用 hasOwnProperty 过滤)
-
使用
for...of
当:- 需要遍历数组的元素值
- 需要遍历其他可迭代对象(Map, Set等)的值
- 需要更简洁的语法且不关心索引/键
-
都不使用时:
- 对于简单的数组遍历,传统的
for
循环有时性能更好 - 需要中断循环时,
forEach
无法使用break
,此时for...of
更合适
- 对于简单的数组遍历,传统的
实际应用示例
示例1:对象属性操作
const user = {id: 1,name: 'John Doe',email: 'john@example.com',isAdmin: false
};// 使用 for...in 将所有布尔属性转换为字符串
for (const key in user) {if (typeof user[key] === 'boolean') {user[key] = user[key].toString();}
}console.log(user); // isAdmin 现在是 "false"
示例2:数组元素处理
const scores = [85, 92, 78, 95, 88];
let total = 0;// 使用 for...of 计算总分
for (const score of scores) {total += score;
}console.log(`平均分: ${total / scores.length}`); // 平均分: 87.6
示例3:结合解构赋值
const users = [{ id: 1, name: 'Alice' },{ id: 2, name: 'Bob' },{ id: 3, name: 'Charlie' }
];// 使用 for...of 和解构
for (const { id, name } of users) {console.log(`用户ID: ${id}, 姓名: ${name}`);
}
总结
for...in
用于遍历对象的可枚举属性,包括原型链上的属性for...of
用于遍历可迭代对象的值,如数组元素、Map/Set 的项等- 对于数组遍历,优先使用
for...of
而不是for...in
- 两种循环各有适用场景,理解它们的差异有助于写出更清晰、更高效的代码
掌握 for...in
和 for...of
的区别和正确用法,将使你在 JavaScript 开发中能够更灵活地处理各种数据结构和对象遍历需求。
相关文章:
JavaScript 中的 for...in 和 for...of 循环详解
在 JavaScript 中,for...in 和 for...of 是两种常用的循环结构,但它们有着不同的用途和行为。很多初学者容易混淆这两者,本文将详细解析它们的区别、适用场景以及注意事项。 目录 for…in 循环 基本用法遍历对象属性注意事项 for…of 循环 …...
AtCoder AT_abc406_c [ABC406C] ~
前言 除了 A 题,唯一一道一遍过的题。 题目大意 我们定义满足以下所有条件的一个长度为 N N N 的序列 A ( A 1 , A 2 , … , A N ) A(A_1,A_2,\dots,A_N) A(A1,A2,…,AN) 为波浪序列: N ≥ 4 N\ge4 N≥4(其实满足后面就必须满足这…...

Spark,连接MySQL数据库,添加数据,读取数据
连接数据库 可以看到shell中我们读取出的数据 在IDEA中打代码如果能输出跟shell中一样的结果即证明连接成功 【出错反思】 像我前面出错的原因就是在打代码时将密码输入错误 添加数据 读取数据就是在上面代码中一起展示了,这里我就不单独说了...
Linux容器技术详解
容器技术基础 什么是容器 容器是一种轻量级的虚拟化技术,它将应用程序及其依赖(库、二进制文件、配置文件等)打包在一个独立的单元中,可以在任何支持容器运行时的环境中一致地运行。 Docker官网:https://www.docker…...

【EDA软件】【联合Modelsim仿真使用方法】
背景 业界EDA工具仿真功能是必备的,例如Vivado自带仿真工具,且无需联合外部仿真工具,例如MoodelSim。 FUXI工具仿真功能需要联合Modelsim,才能实现仿真功能。 方法一:FUXI联合ModelSim 1 添加testbench文件 新建to…...
STM32 __main
STM32开发中__main与用户main()函数的本质区别及工作机制 在STM32开发中,__main和用户定义的main()函数是启动过程中的两个关键节点,分别承担运行时初始化和用户程序入口的职责。以下是它们的核心差异及协作机制: 一、定义与层级差异 __ma…...

【离散化 线段树】P3740 [HAOI2014] 贴海报|普及+
本文涉及知识点 C线段树 [HAOI2014] 贴海报 题目描述 Bytetown 城市要进行市长竞选,所有的选民可以畅所欲言地对竞选市长的候选人发表言论。为了统一管理,城市委员会为选民准备了一个张贴海报的 electoral 墙。 张贴规则如下: electoral…...
Python训练营打卡Day28
浙大疏锦行 DAY 28 类的定义和方法 知识点回顾: 1.类的定义 2.pass占位语句 3.类的初始化方法 4.类的普通方法 5.类的继承:属性的继承、方法的继承 作业 题目1:定义圆(Circle)类 要求: 1.包含属性&#x…...
MODBUS RTU通信协议详解与调试指南
一、MODBUS RTU简介 MODBUS RTU(Remote Terminal Unit)是一种基于串行通信(RS-485/RS-232)的工业标准协议,采用二进制数据格式,具有高效、可靠的特点,广泛应用于PLC、传感器、变频器等工业设备…...

CSP 2024 提高级第一轮(CSP-S 2024)单选题解析
单选题解析 第 1 题 在 Linux 系统中,如果你想显示当前工作目录的路径,应该使用哪个命令?(A) A. pwd B. cd C. ls D. echo 解析:Linux 系统中,pwd命令可以显示当前工作目录的路径。pwd&#x…...

六、绘制图片
文章目录 1.创建一个红色图片2.加载bmp图片3.加载png、jpg图片 前面的几个示例,我们已经展示过如果在Linux系统下使用xlib接口向窗口中绘制文本、线、矩形;并设置文本、线条的颜色。并利用xlib提供的接口结合事件处理机制完成了一个自绘按钮控件功能。有…...

Java 面向对象详解和JVM底层内存分析
先关注、点赞再看、人生灿烂!!!(谢谢) 神速熟悉面向对象 表格结构和类结构 我们在现实生活中,思考问题、发现问题、处理问题,往往都会用“表格”作为工具。实际上,“表格思维”就是…...

深度学习---知识蒸馏(Knowledge Distillation, KD)
一、知识蒸馏的本质与起源 定义: 知识蒸馏是一种模型压缩与迁移技术,通过将复杂高性能的教师模型(Teacher Model)所学的“知识”迁移到轻量级的学生模型(Student Model),使学生模型在参数量和计…...

基于CNN卷积神经网络的带频偏QPSK调制信号检测识别算法matlab仿真
目录 1.算法运行效果图预览 2.算法运行软件版本 3.部分核心程序 4.算法理论概述 5.算法完整程序工程 1.算法运行效果图预览 (完整程序运行后无水印) 2.算法运行软件版本 matlab2024b 3.部分核心程序 (完整版代码包含详细中文注释和操作步骤视频)…...

【DAY21】 常见的降维算法
内容来自浙大疏锦行python打卡训练营 浙大疏锦行 目录 PCA主成分分析 t-sne降维 线性判别分析 (Linear Discriminant Analysis, LDA) 作业: 什么时候用到降维 降维的主要应用场景 知识点回顾: PCA主成分分析t-sne降维LDA线性判别 通常情况下,…...
PostGIS实现栅格数据入库-raster2pgsql
raster2pgsql使用与最佳实践 一、工具概述 raster2pgsql是PostGIS提供的命令行工具,用于将GDAL支持的栅格格式(如GeoTIFF、JPEG、PNG等)导入PostgreSQL数据库,支持批量加载、分块切片、创建空间索引及金字塔概览,是栅格数据入库的核心工具。 二、核心功能与典型用法 1…...

校园社区小程序源码解析
基于ThinkPHP、FastAdmin和UniApp开发的校园社区小程序源码,旨在为校园内的学生和教职员工提供一个便捷的在线交流和服务平台。 该小程序前端采用UniApp进行开发,具有良好的跨平台兼容性,可以轻松发布到iOS和Android平台。同时,后…...

第6章:文件权限
一、文件权限概述 Linux为了保证系统中每个文件的安全,引入了文件权限机制。针对于系统中的每一个文件Linux都可以提供精确的权限控制。它可以做到不同的用户对同一个文件具有不同的操作权利。而通常这个权利包括以下3个: 读的权利(Read&…...
使用 Python 连接 Oracle 23ai 数据库完整指南
方法一:使用 oracledb 官方驱动(推荐) Oracle 官方维护的 oracledb 驱动(原 cx_Oracle)是最新推荐方案,支持 Thin/Thick 两种模式。 1. 环境准备 pip install oracledb2. 完整示例代码 import oracledb import getpass from typing import Unionclass Oracle23aiConn…...

C语言| 指针变量的定义
C语言| 指针的优点-CSDN博客 * 表示“指向”,为了说明指针变量和它所指向的变量之间的联系。 int * i;//表示指针变量i里面存放的地址,所指向的存储单元里的【数据】。 【指针变量的定义】 C语言规定所有变量,在使用前必须先定…...
HTML 中的 input 标签详解
HTML 中的 input 标签详解 一、基础概念 1. 定义与作用 HTML 中的 <input> 标签是表单元素的核心组件,用于创建各种用户输入字段。作为一个空标签(没有闭合标签),它通过 type 属性来决定呈现何种输入控件,是实…...
Python 在自动驾驶数据标签中的应用:如何让 AI 读懂道路?
Python 在自动驾驶数据标签中的应用:如何让 AI 读懂道路? 在自动驾驶系统中,数据就是生命线。不管是摄像头、激光雷达还是雷达传感器,这些设备每天都能产生 海量数据,但如果这些数据没有被正确标注,它们对 AI 来说毫无意义。那么,如何让自动驾驶系统准确理解道路环境呢…...
微信小程序之按钮短时间内被多次点击问题
做项目的时候碰到这个问题,按钮的功能做好了,但是总会出现按的太快,出现不可预料的问题。 解决方法之一:借助函数节流来实现 1、创建一个工具包(throttle.js),通过封装一个高阶函数,对函数的执…...
动态规划(3)学习方法论:构建思维模型
引言 动态规划是算法领域中一个强大而优雅的解题方法,但对于许多学习者来说,它也是最难以掌握的算法范式之一。与贪心算法或分治法等直观的算法相比,动态规划往往需要更抽象的思维和更系统的学习方法。在前两篇文章中,我们介绍了动态规划的基础概念、原理以及问题建模与状…...
两个电机由同一个控制器控制,其中一个电机发生堵转时,另一个电机的电流会变大,是发生了倒灌现象吗?电流倒灌产生的机理是什么?
当两个电机由同一个控制器驱动,且其中一个电机发生堵转时,另一个电机的电流确实可能异常增大,但这不一定是典型的“倒灌现象”,而更可能是由于共母线电压波动或能量回馈导致的。以下是具体分析: 1. 现象是否属于“电流…...
Java 方法向 Redis 里操作字符串有什么需要注意的?
在 Java 开发中,Redis 作为高性能的键值存储数据库,常被用于缓存数据、处理高并发场景等。当我们使用 Java 方法向 Redis 中操作字符串类型数据时,有许多关键要点需要格外注意。这些要点不仅关系到代码的正确性和性能,还影响着整个…...
ECMAScript 2018(ES2018):异步编程与正则表达式的深度进化
1.版本背景与发布 发布时间:2018年6月,由ECMA International正式发布,标准编号为ECMA-262 9th Edition。历史意义:作为ES6之后的第三次年度更新,ES2018聚焦于异步编程、正则表达式和对象操作的标准化,推动…...

IntelliJ IDEA给Controller、Service、Mapper不同文件设置不同的文件头注释模板、Velocity模板引擎
通过在 IntelliJ IDEA 中的 “Includes” 部分添加多个文件头模板,并在 “Files” 模板中利用这些包含来实现不同类型文件的注释。以下是为 Controller、Service、Mapper 文件设置不同文件头的完整示例: 1. 设置 Includes 文件头模板 File > Settin…...
从零开始认识 Node.js:异步非阻塞的魅力
Node.js 是一个基于 Chrome V8 引擎 的 JavaScript 运行时环境,用于在服务器端运行 JavaScript 代码。它的设计目标是让开发者能够用 JavaScript 构建高性能、可扩展的网络应用。以下是关于 Node.js 的详细介绍: 1. 核心特点 事件驱动与非阻塞 I/O&…...
【C语言练习】046. 编写插入排序算法
046. 编写插入排序算法 046. 编写插入排序算法C语言实现插入排序代码说明示例运行输入:输出:插入排序的特点一、插入排序的适用场景二、C语言代码示例及分步讲解代码实现代码解析三、示例执行过程四、性能分析五、总结046. 编写插入排序算法 插入排序(Insertion Sort)是一…...