목차
JavaScript의 for 루프에 대한 심층적인 이해
간단한 for 루프
for-in
forEach
for-of
웹 프론트엔드 JS 튜토리얼 js의 네 가지 for 루프

js의 네 가지 for 루프

Dec 08, 2017 pm 04:09 PM
javascript 주기

JavaScript의 for 루프에 대한 심층적인 이해

ECMAScript5(줄여서 ES5)에는 세 가지 종류의 for 루프가 있습니다. 즉,

2015년 6월에 출시된 ECMAScript6(줄여서 ES6)에는 새로운 루프가 추가되었습니다. , 즉 :

4가지 유형의 for 루프를 살펴보겠습니다.

간단한 for 루프

가장 일반적인 작성 방법을 살펴보겠습니다.

const arr = [1, 2, 3];for(let i = 0; i < arr.length; i++) {    console.log(arr[i]);
}
로그인 후 복사

루프 중에 배열의 길이가 변경되지 않으면 배열의 길이를 변수에 저장해야 합니다. 더 나은 결과 효율성을 위해 다음은 향상된 작성 방법입니다.

const arr = [1, 2, 3];for(let i = 0, len = arr.length; i < len; i++) {
    console.log(arr[i]);
}
로그인 후 복사

for-in

일반적으로 for-in을 사용하여 배열의 내용을 탐색할 수 있습니다.

const arr = [1, 2, 3];let index;for(index in arr) {    console.log("arr[" + index + "] = " + arr[index]);
}
로그인 후 복사

일반적으로, 실행 결과는 다음과 같습니다.

arr[0] = 1
arr[1] = 2
arr[2] = 3
로그인 후 복사

하지만 이렇게 하면 종종 문제가 발생합니다.

for-in

for-in 루프에 대한 진실은 배열의 인덱스가 아닌 객체의 속성을 순회합니다. 따라서 for-in으로 순회하는 객체는 배열에 국한되지 않고 객체도 순회할 수 있습니다. 예는 다음과 같습니다:

const person = {
    fname: "san",
    lname: "zhang",
    age: 99};let info;for(info in person) {    console.log("person[" + info + "] = " + person[info]);
}
로그인 후 복사

결과는 다음과 같습니다:

person[fname] = san
person[lname] = zhang
person[age] = 99
로그인 후 복사

for-in이 속성을 순회하는 순서는 결정되지 않습니다. 즉, 출력 결과의 순서는 아무 관련이 없습니다. 객체의 속성 순서나 속성의 알파벳 순서도 중요하지 않습니다.

Array의 진실

Array는 Javascript의 객체이고 Array의 인덱스는 속성 이름입니다. 실제로 Javascript의 "배열"은 다소 오해의 소지가 있습니다. Javascript의 배열은 대부분의 다른 언어의 배열과 다릅니다. 첫째, Javascript의 Array는 메모리에서 연속적이지 않습니다. 둘째, Array의 인덱스는 오프셋을 참조하지 않습니다. 실제로 Array의 인덱스는 Number형이 아니지만, 문자열 유형입니다. arr[0]을 올바르게 사용할 수 있는 이유는 언어가 자동으로 Number 유형의 0을 String 유형의 "0"으로 변환할 수 있기 때문입니다. 따라서 Javascript에는 배열 인덱스가 없고 "0", "1" 등과 같은 속성만 있습니다. 흥미롭게도 모든 Array 객체에는 길이 속성이 있어 다른 언어의 배열처럼 동작합니다. 그런데 Array 객체를 순회할 때 길이 속성이 출력되지 않는 이유는 무엇입니까? for-in은 "열거 가능한 속성"에 대해서만 반복할 수 있기 때문입니다. 실제로 length는 열거 불가능한 속성입니다. 배열 객체에는 열거할 수 없는 다른 속성이 많이 있습니다.

이제 다시 돌아가 for-in을 사용하여 배열을 반복하는 예제를 살펴보겠습니다. 배열을 순회하는 이전 예제를 수정해 보겠습니다.

const arr = [1, 2, 3];
arr.name = "Hello world";let index;for(index in arr) {    console.log("arr[" + index + "] = " + arr[index]);
}
로그인 후 복사

실행 결과는 다음과 같습니다.

arr[0] = 1
arr[1] = 2
arr[2] = 3
arr[name] = Hello world
로그인 후 복사

for-in이 표시됩니다. for-in이 "인덱스"뿐만 아니라 개체의 모든 속성을 반복하기 때문에 새로운 "이름" 속성이 추가되었습니다. 동시에 여기서 출력되는 인덱스 값, 즉 "0", "1", "2"는 속성으로 출력되기 때문에 Number 타입이 아닌 String 타입이라는 점에 유의해야 한다. , 인덱스가 아닙니다. 이는 Array 객체에 새 속성을 추가하지 않고 배열의 내용만 출력할 수 있다는 의미입니까? 대답은 '아니요'입니다. 왜냐하면 for-in은 배열 자체의 속성뿐만 아니라 배열 프로토타입 체인의 모든 열거 가능한 속성도 탐색합니다. 아래 예를 살펴보겠습니다.

Array.prototype.fatherName = "Father";const arr = [1, 2, 3];
arr.name = "Hello world";let index;for(index in arr) {    console.log("arr[" + index + "] = " + arr[index]);
}
로그인 후 복사

실행 결과는 다음과 같습니다.

arr[0] = 1
arr[1] = 2
arr[2] = 3
arr[name] = Hello world
arr[fatherName] = Father
로그인 후 복사

이것을 작성하면 for-in이 배열의 요소를 순회하는 데 적합하지 않고 객체의 속성을 순회하는 데 더 적합하다는 것을 알 수 있습니다. 그것은 창조의 원래 의도를 사용합니다. 한 가지 예외, 즉 희소 배열이 있습니다. 다음 예를 고려하십시오.

let key;const arr = [];
arr[0] = "a";
arr[100] = "b";
arr[10000] = "c";for(key in arr) {    if(arr.hasOwnProperty(key)  &&    
        /^0$|^[1-9]\d*$/.test(key) &&    
        key <= 4294967294               
        ) {        console.log(arr[key]);
    }
}
로그인 후 복사

for-in은 기존 엔터티만 순회합니다. 위 예에서 for-in은 각각 "0", "100" 및 "10000" 속성을 가진 요소를 순회합니다. for 루프는 10001번 반복됩니다). 따라서 올바르게 수행되는 한 for-in은 배열의 요소를 탐색하는 데 큰 역할을 할 수도 있습니다.

작업 중복을 피하기 위해 위의 코드를 래핑할 수 있습니다.

function arrayHasOwnIndex(array, prop) {    return array.hasOwnProperty(prop) && 
        /^0$|^[1-9]\d*$/.test(prop) && 
        prop <= 4294967294; // 2^32 - 2}
로그인 후 복사

사용 예는 다음과 같습니다.

for (let key in arr) {    if (arrayHasOwnIndex(arr, key)) {
        console.log(arr[key]);
    }
}
로그인 후 복사

for-in 성능

正如上面所说,每次迭代操作会同时搜索实例或者原型属性, for-in 循环的每次迭代都会产生更多开销,因此要比其他循环类型慢,一般速度为其他类型循环的 1/7。因此,除非明确需要迭代一个属性数量未知的对象,否则应避免使用 for-in 循环。如果需要遍历一个数量有限的已知属性列表,使用其他循环会更快,比如下面的例子:

const obj = {    "prop1": "value1",    "prop2": "value2"};const props = ["prop1", "prop2"];for(let i = 0; i < props.length; i++) {    console.log(obj[props[i]]);
}
로그인 후 복사

上面代码中,将对象的属性都存入一个数组中,相对于 for-in 查找每一个属性,该代码只关注给定的属性,节省了循环的开销和时间。

forEach

在 ES5 中,引入了新的循环,即 forEach 循环。

const arr = [1, 2, 3];
arr.forEach((data) => {    console.log(data);
});
로그인 후 복사

运行结果:

1
2
3
로그인 후 복사

forEach 方法为数组中含有有效值的每一项执行一次 callback 函数,那些已删除(使用 delete 方法等情况)或者从未赋值的项将被跳过(不包括那些值为 undefined 或 null 的项)。 callback 函数会被依次传入三个参数:

  • 数组当前项的值;

  • 数组当前项的索引;

  • 数组对象本身;

需要注意的是,forEach 遍历的范围在第一次调用 callback 前就会确定。调用forEach 后添加到数组中的项不会被 callback 访问到。如果已经存在的值被改变,则传递给 callback 的值是 forEach 遍历到他们那一刻的值。已删除的项不会被遍历到。

const arr = [];
arr[0] = "a";
arr[3] = "b";
arr[10] = "c";
arr.name = "Hello world";
arr.forEach((data, index, array) => {
    console.log(data, index, array);
});
로그인 후 복사

运行结果:

a 0 ["a", 3: "b", 10: "c", name: "Hello world"]
b 3 ["a", 3: "b", 10: "c", name: "Hello world"]
c 10 ["a", 3: "b", 10: "c", name: "Hello world"]
로그인 후 복사

这里的 index 是 Number 类型,并且也不会像 for-in 一样遍历原型链上的属性。

所以,使用 forEach 时,我们不需要专门地声明 index 和遍历的元素,因为这些都作为回调函数的参数。

另外,forEach 将会遍历数组中的所有元素,但是 ES5 定义了一些其他有用的方法,下面是一部分:

  • every: 循环在第一次 return false 后返回

  • some: 循环在第一次 return true 后返回

  • filter: 返回一个新的数组,该数组内的元素满足回调函数

  • map: 将原数组中的元素处理后再返回

  • reduce: 对数组中的元素依次处理,将上次处理结果作为下次处理的输入,最后得到最终结果。

forEach 性能

首先感谢@papa pa的提醒,才发现我之前的理解有错误。

大家可以看 jsPerf ,在不同浏览器下测试的结果都是 forEach 的速度不如 for。如果大家把测试代码放在控制台的话,可能会得到不一样的结果,主要原因是控制台的执行环境与真实的代码执行环境有所区别。

for-of

先来看个例子:

const arr = ['a', 'b', 'c'];for(let data of arr) {    console.log(data);
}
로그인 후 복사

运行结果是:

ab
c
로그인 후 복사

为什么要引进 for-of?

要回答这个问题,我们先来看看ES6之前的 3 种 for 循环有什么缺陷:

  • forEach 不能 break 和 return;

  • for-in 缺点更加明显,它不仅遍历数组中的元素,还会遍历自定义的属性,甚至原型链上的属性都被访问到。而且,遍历数组元素的顺序可能是随机的。

所以,鉴于以上种种缺陷,我们需要改进原先的 for 循环。但 ES6 不会破坏你已经写好的 JS 代码。目前,成千上万的 Web 网站依赖 for-in 循环,其中一些网站甚至将其用于数组遍历。如果想通过修正 for-in 循环增加数组遍历支持会让这一切变得更加混乱,因此,标准委员会在 ES6 中增加了一种新的循环语法来解决目前的问题,即 for-of 。

那 for-of 到底可以干什么呢?

  • 跟 forEach 相比,可以正确响应 break, continue, return。

  • for-of 循环不仅支持数组,还支持大多数类数组对象,例如 DOM nodelist 对象。

  • for-of 循环也支持字符串遍历,它将字符串视为一系列 Unicode 字符来进行遍历。

  • for-of 也支持 Map 和 Set (两者均为 ES6 中新增的类型)对象遍历。

总结一下,for-of 循环有以下几个特征:

  • 这是最简洁、最直接的遍历数组元素的语法。

  • 这个方法避开了 for-in 循环的所有缺陷。

  • 与 forEach 不同的是,它可以正确响应 break、continue 和 return 语句。

  • 其不仅可以遍历数组,还可以遍历类数组对象和其他可迭代对象。

但需要注意的是,for-of循环不支持普通对象,但如果你想迭代一个对象的属性,你可以用 for-in 循环(这也是它的本职工作)。

最后要说的是,ES6 引进的另一个方式也能实现遍历数组的值,那就是 Iterator。上个例子:

const arr = [&#39;a&#39;, &#39;b&#39;, &#39;c&#39;];
const iter = arr[Symbol.iterator]();
iter.next() // { value: &#39;a&#39;, done: false }
iter.next() // { value: &#39;b&#39;, done: false }
iter.next() // { value: &#39;c&#39;, done: false }
iter.next() // { value: undefined, done: true }
로그인 후 복사

相关推荐:

JavaScript中for循环以及if判断语句详解

for循环写法进阶

php中for循环的扩展用法实例详解

위 내용은 js의 네 가지 for 루프의 상세 내용입니다. 자세한 내용은 PHP 중국어 웹사이트의 기타 관련 기사를 참조하세요!

본 웹사이트의 성명
본 글의 내용은 네티즌들의 자발적인 기여로 작성되었으며, 저작권은 원저작자에게 있습니다. 본 사이트는 이에 상응하는 법적 책임을 지지 않습니다. 표절이나 침해가 의심되는 콘텐츠를 발견한 경우 admin@php.cn으로 문의하세요.

핫 AI 도구

Undresser.AI Undress

Undresser.AI Undress

사실적인 누드 사진을 만들기 위한 AI 기반 앱

AI Clothes Remover

AI Clothes Remover

사진에서 옷을 제거하는 온라인 AI 도구입니다.

Undress AI Tool

Undress AI Tool

무료로 이미지를 벗다

Clothoff.io

Clothoff.io

AI 옷 제거제

AI Hentai Generator

AI Hentai Generator

AI Hentai를 무료로 생성하십시오.

인기 기사

R.E.P.O. 에너지 결정과 그들이하는 일 (노란색 크리스탈)
1 몇 달 전 By 尊渡假赌尊渡假赌尊渡假赌
R.E.P.O. 최고의 그래픽 설정
1 몇 달 전 By 尊渡假赌尊渡假赌尊渡假赌
Will R.E.P.O. 크로스 플레이가 있습니까?
1 몇 달 전 By 尊渡假赌尊渡假赌尊渡假赌

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

사용하기 쉬운 무료 코드 편집기

SublimeText3 중국어 버전

SublimeText3 중국어 버전

중국어 버전, 사용하기 매우 쉽습니다.

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

신 수준의 코드 편집 소프트웨어(SublimeText3)

WebSocket과 JavaScript를 사용하여 온라인 음성 인식 시스템을 구현하는 방법 WebSocket과 JavaScript를 사용하여 온라인 음성 인식 시스템을 구현하는 방법 Dec 17, 2023 pm 02:54 PM

WebSocket 및 JavaScript를 사용하여 온라인 음성 인식 시스템을 구현하는 방법 소개: 지속적인 기술 개발로 음성 인식 기술은 인공 지능 분야의 중요한 부분이 되었습니다. WebSocket과 JavaScript를 기반으로 한 온라인 음성 인식 시스템은 낮은 대기 시간, 실시간, 크로스 플랫폼이라는 특징을 갖고 있으며 널리 사용되는 솔루션이 되었습니다. 이 기사에서는 WebSocket과 JavaScript를 사용하여 온라인 음성 인식 시스템을 구현하는 방법을 소개합니다.

WebSocket 및 JavaScript: 실시간 모니터링 시스템 구현을 위한 핵심 기술 WebSocket 및 JavaScript: 실시간 모니터링 시스템 구현을 위한 핵심 기술 Dec 17, 2023 pm 05:30 PM

WebSocket과 JavaScript: 실시간 모니터링 시스템 구현을 위한 핵심 기술 서론: 인터넷 기술의 급속한 발전과 함께 실시간 모니터링 시스템이 다양한 분야에서 널리 활용되고 있다. 실시간 모니터링을 구현하는 핵심 기술 중 하나는 WebSocket과 JavaScript의 조합입니다. 이 기사에서는 실시간 모니터링 시스템에서 WebSocket 및 JavaScript의 적용을 소개하고 코드 예제를 제공하며 구현 원칙을 자세히 설명합니다. 1. 웹소켓 기술

JavaScript 및 WebSocket을 사용하여 실시간 온라인 주문 시스템을 구현하는 방법 JavaScript 및 WebSocket을 사용하여 실시간 온라인 주문 시스템을 구현하는 방법 Dec 17, 2023 pm 12:09 PM

JavaScript 및 WebSocket을 사용하여 실시간 온라인 주문 시스템을 구현하는 방법 소개: 인터넷의 대중화와 기술의 발전으로 점점 더 많은 레스토랑에서 온라인 주문 서비스를 제공하기 시작했습니다. 실시간 온라인 주문 시스템을 구현하기 위해 JavaScript 및 WebSocket 기술을 사용할 수 있습니다. WebSocket은 TCP 프로토콜을 기반으로 하는 전이중 통신 프로토콜로 클라이언트와 서버 간의 실시간 양방향 통신을 실현할 수 있습니다. 실시간 온라인 주문 시스템에서는 사용자가 요리를 선택하고 주문을 하면

WebSocket과 JavaScript를 사용하여 온라인 예약 시스템을 구현하는 방법 WebSocket과 JavaScript를 사용하여 온라인 예약 시스템을 구현하는 방법 Dec 17, 2023 am 09:39 AM

WebSocket과 JavaScript를 사용하여 온라인 예약 시스템을 구현하는 방법 오늘날의 디지털 시대에는 점점 더 많은 기업과 서비스에서 온라인 예약 기능을 제공해야 합니다. 효율적인 실시간 온라인 예약 시스템을 구현하는 것이 중요합니다. 이 기사에서는 WebSocket과 JavaScript를 사용하여 온라인 예약 시스템을 구현하는 방법을 소개하고 구체적인 코드 예제를 제공합니다. 1. WebSocket이란 무엇입니까? WebSocket은 단일 TCP 연결의 전이중 방식입니다.

JavaScript와 WebSocket: 효율적인 실시간 일기예보 시스템 구축 JavaScript와 WebSocket: 효율적인 실시간 일기예보 시스템 구축 Dec 17, 2023 pm 05:13 PM

JavaScript 및 WebSocket: 효율적인 실시간 일기 예보 시스템 구축 소개: 오늘날 일기 예보의 정확성은 일상 생활과 의사 결정에 매우 중요합니다. 기술이 발전함에 따라 우리는 날씨 데이터를 실시간으로 획득함으로써 보다 정확하고 신뢰할 수 있는 일기예보를 제공할 수 있습니다. 이 기사에서는 JavaScript 및 WebSocket 기술을 사용하여 효율적인 실시간 일기 예보 시스템을 구축하는 방법을 알아봅니다. 이 문서에서는 특정 코드 예제를 통해 구현 프로세스를 보여줍니다. 우리

간단한 JavaScript 튜토리얼: HTTP 상태 코드를 얻는 방법 간단한 JavaScript 튜토리얼: HTTP 상태 코드를 얻는 방법 Jan 05, 2024 pm 06:08 PM

JavaScript 튜토리얼: HTTP 상태 코드를 얻는 방법, 특정 코드 예제가 필요합니다. 서문: 웹 개발에서는 서버와의 데이터 상호 작용이 종종 포함됩니다. 서버와 통신할 때 반환된 HTTP 상태 코드를 가져와서 작업의 성공 여부를 확인하고 다양한 상태 코드에 따라 해당 처리를 수행해야 하는 경우가 많습니다. 이 기사에서는 JavaScript를 사용하여 HTTP 상태 코드를 얻는 방법과 몇 가지 실용적인 코드 예제를 제공합니다. XMLHttpRequest 사용

람다 표현식이 루프에서 벗어남 람다 표현식이 루프에서 벗어남 Feb 20, 2024 am 08:47 AM

람다 표현식은 루프에서 벗어나기 때문에 특정 코드 예제가 필요합니다. 프로그래밍에서 루프 구조는 자주 사용되는 중요한 구문입니다. 그러나 특정 상황에서는 현재 루프 반복을 종료하는 대신 루프 본문 내에서 특정 조건이 충족될 때 전체 루프를 중단하고 싶을 수도 있습니다. 이때 람다 표현식의 특징은 루프에서 벗어나는 목표를 달성하는 데 도움이 될 수 있습니다. 람다 표현식은 내부적으로 간단한 함수 논리를 정의할 수 있는 익명 함수를 선언하는 방법입니다. 일반적인 함수 선언과는 다릅니다.

JavaScript 및 WebSocket: 효율적인 실시간 이미지 처리 시스템 구축 JavaScript 및 WebSocket: 효율적인 실시간 이미지 처리 시스템 구축 Dec 17, 2023 am 08:41 AM

JavaScript는 웹 개발에 널리 사용되는 프로그래밍 언어인 반면 WebSocket은 실시간 통신에 사용되는 네트워크 프로토콜입니다. 두 가지의 강력한 기능을 결합하면 효율적인 실시간 영상 처리 시스템을 만들 수 있습니다. 이 기사에서는 JavaScript와 WebSocket을 사용하여 이 시스템을 구현하는 방법을 소개하고 구체적인 코드 예제를 제공합니다. 첫째, 실시간 영상처리 시스템의 요구사항과 목표를 명확히 할 필요가 있다. 실시간 이미지 데이터를 수집할 수 있는 카메라 장치가 있다고 가정해 보겠습니다.

See all articles