목차
전제 지식
Promise 마이크로태스크와 관련된 API는 무엇인가요?
这些微任务何时被加入微任务队列?
同一个 then,不同的微任务执行
初级
中级
高级
根据 Promise A+ 规范
根据 ECMA - 262 规范
最后
웹 프론트엔드 JS 튜토리얼 진정으로 이해해야 할 약속

진정으로 이해해야 할 약속

Sep 07, 2020 pm 01:30 PM
promise

진정으로 이해해야 할 약속

관련 학습 권장 사항: javascript 학습 튜토리얼

약속 API에 관해서는 누구나 능숙하게 사용할 수 있어야 하지만, 여전히 마이크로 태스크와 관련된 지식 사각지대가 있을 수 있습니다.

전제 지식

본문을 시작하기 전에 이 기사에 포함된 일부 내용의 분위기를 미리 설정해 보겠습니다.

Promise 마이크로태스크와 관련된 API는 무엇인가요?

Promise에서는 then, catchfinally와 같이 상태 변경이 관련된 후에 실행해야 하는 콜백만 마이크로 작업으로 간주됩니다. 다른 모든 코드 실행은 매크로 작업(동기 실행)입니다. thencatchfinally ,其他所有的代码执行都是宏任务(同步执行)。

진정으로 이해해야 할 약속

上图中蓝色为同步执行,黄色为异步执行(丢到微任务队列中)。

这些微任务何时被加入微任务队列?

这个问题我们根据 ecma 规范来看:

  • 如果此时 Promise 状态为 pending,那么成功或失败的回调会分别被加入至 [[PromiseFulfillReactions]][[PromiseRejectReactions]] 中。如果你看过手写 Promise 的代码的话,应该能发现有两个数组存储这些回调函数。

  • 如果此时 Promise 状态为非 pending 时,回调会成为 Promise Jobs,也就是微任务。

了解完以上知识后,正片开始。

同一个 then,不同的微任务执行

初级

Promise.resolve()
  .then(() => {    console.log("then1");    Promise.resolve().then(() => {      console.log("then1-1");
    });
  })
  .then(() => {    console.log("then2");
  });复制代码
로그인 후 복사

以上代码大家应该都能得出正确的答案:then1 → then1-1 → then2

虽然 then 是同步执行,并且状态也已经变更。但这并不代表每次遇到 then 时我们都需要把它的回调丢入微任务队列中,而是等待 then 的回调执行完毕后再根据情况执行对应操作。

基于此,我们可以得出第一个结论:链式调用中,只有前一个 then 的回调执行完毕后,跟着的 then 中的回调才会被加入至微任务队列。

中级

大家都知道了 Promise resolve 后,跟着的 then 中的回调会马上进入微任务队列。

那么以下代码你认为的输出会是什么?

let p = Promise.resolve();

p.then(() => {  console.log("then1");  Promise.resolve().then(() => {    console.log("then1-1");
  });
}).then(() => {  console.log("then1-2");
});

p.then(() => {  console.log("then2");
}); 
复制代码
로그인 후 복사

按照一开始的认知我们不难得出 then2 会在 then1-1 后输出,但是实际情况却是相反的。

基于此我们得出第二个结论:每个链式调用的开端会首先依次进入微任务队列。

接下来我们换个写法:

let p = Promise.resolve().then(() => {  console.log("then1");  Promise.resolve().then(() => {    console.log("then1-1");
  });
}).then(() => {  console.log("then2");
});

p.then(() => {  console.log("then3");
});复制代码
로그인 후 복사

上述代码其实有个陷阱,then 每次都会返回一个新的 Promise,此时的 p 已经不是 Promise.resolve() 生成的,而是最后一个 then 生成的,因此 then3 应该是在 then2 后打印出来的。

顺便我们也可以把之前得出的结论优化为:同一个 Promise 的每个链式调用的开端会首先依次进入微任务队列。

高级

以下大家可以猜猜 then1-2 会在何时打印出来?

Promise.resolve()
  .then(() => {    console.log("then1");    Promise.resolve()
      .then(() => {        console.log("then1-1");        return 1;
      })
      .then(() => {        console.log("then1-2");
      });
  })
  .then(() => {    console.log("then2");
  })
  .then(() => {    console.log("then3");
  })
  .then(() => {    console.log("then4");
  });复制代码
로그인 후 복사

这题肯定是简单的,记住第一个结论就能得出答案,以下是解析:

  • 第一次 resolve 后第一个 then 的回调进入微任务队列并执行,打印 then1

  • 第二次 resolve 后内部第一个 then 的回调进入微任务队列,此时外部第一个 then 的回调全部执行完毕,需要将外部的第二个 then 回调也插入微任务队列。

  • 执行微任务,打印 then1-1then2,然后分别再将之后 then 中的回调插入微任务队列

  • 执行微任务,打印 then1-2then3 ,之后的内容就不一一说明了

接下来我们把 return 1 修改一下,结果可就大不相同啦:

Promise.resolve()
  .then(() => {    console.log("then1");    Promise.resolve()
      .then(() => {        console.log("then1-1");        return Promise.resolve();
      })
      .then(() => {        console.log("then1-2");
      });
  })
  .then(() => {    console.log("then2");
  })
  .then(() => {    console.log("then3");
  })
  .then(() => {    console.log("then4");
  });复制代码
로그인 후 복사

当我们 return Promise.resolve() 时,你猜猜 then1-2 会何时打印了?

答案是最后一个才被打印出来。

为什么在 then 中分别 return<그림>진정으로 이해해야 할 약속

위 그림에서 파란색은 동기 실행을 나타내고 노란색은 비동기 실행(마이크로태스크 대기열에 던져짐)을 나타냅니다. 🎜

이러한 마이크로태스크는 언제 마이크로태스크 대기열에 추가되나요? 🎜🎜ecma 사양에 따라 이 문제를 살펴보겠습니다. 🎜
  • 🎜현재 Promise 상태가 보류 중이면 성공 또는 실패한 콜백이 [[PromiseFulfillReactions]]<에 추가됩니다. /code> 및 <code>[[PromiseRejectReactions]]. 손으로 작성한 Promise 코드를 살펴보면 이러한 콜백 함수를 저장하는 배열이 두 개 있다는 것을 알 수 있습니다. 🎜
  • 🎜현재 Promise 상태가 보류 중이 아닌 경우 콜백은 마이크로태스크인 Promise Job이 됩니다. 🎜
🎜위의 지식을 이해한 후 본편이 시작됩니다. 🎜

같으면 다른 마이크로 작업 실행🎜

초급🎜
if (x instanceof MyPromise) {  if (x.currentState === PENDING) {
  } else {
    x.then(resolve, reject);
  }  return;
}复制代码
로그인 후 복사
로그인 후 복사
🎜누구나 할 수 있어야 합니다. 위 코드 정답을 제시하세요: then1 → then1-1 → then2. 🎜🎜then이 동기적으로 실행되어 상태가 변경되었지만. 하지만 이것이 then을 만날 때마다 콜백을 마이크로태스크 대기열에 던져야 한다는 의미는 아닙니다. 대신 then의 콜백이 완료될 때까지 기다려야 합니다. 그런 다음 상황에 따라 해당 작업을 수행하십시오. 🎜🎜이를 바탕으로 첫 번째 결론을 내릴 수 있습니다. 체인 호출에서는 이전 then 콜백이 실행된 후에만 다음 만 수행됩니다. Strong>then의 콜백이 마이크로태스크 대기열에 추가됩니다. 🎜

중급 🎜🎜모두가 Promise 해결 후에 다음 then의 콜백이 즉시 Enter라는 것을 알고 있습니다. 마이크로태스크 큐. 🎜🎜그러면 다음 코드의 출력은 어떻게 될까요? 🎜rrreee🎜초기 이해에 따르면 then2then1-1 다음에 출력될 것이라고 결론을 내리는 것은 어렵지 않지만 실제 상황은 그 반대입니다. 🎜🎜이를 바탕으로 우리는 두 번째 결론을 내립니다.각 체인 호출의 시작은 먼저 마이크로태스크 대기열에 순차적으로 들어갑니다. 🎜🎜다음으로 작성 방식을 바꿔 보겠습니다. 🎜rrreee🎜위 코드에는 실제로 트랩이 있습니다. then은 이때마다 p를 반환합니다. 더 이상 Promise.resolve()에 의해 생성되지 않고 마지막 then에 의해 생성되므로 then3은 < code>then2가 출력됩니다. 🎜🎜그런데 이전에 도달한 결론을 최적화할 수도 있습니다.동일한 Promise의 각 체인 호출이 시작되면 먼저 마이크로태스크 대기열에 순차적으로 들어갑니다. 🎜

Advanced🎜🎜 then1-2가 언제 인쇄될지 추측할 수 있나요? 🎜rrreee🎜이 질문은 확실히 간단합니다. 첫 번째 결론만 기억하면 답을 얻을 수 있습니다. 🎜
  • 🎜첫 번째 resolve 코드 콜백>then이 마이크로태스크 대기열에 들어가서 실행되어 then1🎜
  • 🎜두 번째 resolve 이후 첫 번째 내부 대기열을 인쇄합니다. then의 콜백은 마이크로태스크 큐에 들어갑니다. 이때 첫 번째 외부 then의 콜백이 모두 실행되었고, 두 번째 외부 then이 실행되었습니다. > 실행이 필요합니다. 콜백도 마이크로태스크 대기열에 삽입됩니다. 🎜
  • 🎜마이크로 작업을 실행하고 then1-1then2를 인쇄한 다음 각각 then에서 콜백을 호출합니다. the microtask queue🎜
  • 🎜microtask를 실행하고 then1-2then3를 인쇄하세요. 다음 내용은 하나씩 설명하지 않습니다🎜
🎜다음으로 return 1을 수정했는데 결과는 상당히 다릅니다. 🎜rrreee🎜 return Promise.resolve() 를 사용하면 >then1-2가 인쇄됩니까? 🎜🎜정답은 마지막에 인쇄된 답변입니다. 🎜🎜 thenreturn이 다른데 왜 마이크로태스크의 실행 순서가 이렇게 크게 바뀌나요? 다음은 저자의 분석이다. 🎜

PS:then 返回一个新的 Promise,并且会用这个 Promise 去 resolve 返回值,这个概念需要大家先了解一下。

根据 Promise A+ 规范

根据规范 2.3.2,如果 resolve 了一个 Promise,需要为其加上一个 thenresolve

if (x instanceof MyPromise) {  if (x.currentState === PENDING) {
  } else {
    x.then(resolve, reject);
  }  return;
}复制代码
로그인 후 복사
로그인 후 복사

上述代码节选自手写 Promise 实现。

那么根据 A+ 规范来说,如果我们在 then 中返回了 Promise.resolve 的话会多入队一次微任务,但是这个结论还是与实际不符的,因此我们还需要寻找其他权威的文档。

根据 ECMA - 262 规范

根据规范 25.6.1.3.2,当 Promise resolve 了一个 Promise 时,会产生一个NewPromiseResolveThenableJob,这是属于 Promise Jobs 中的一种,也就是微任务。

This Job uses the supplied thenable and its then method to resolve the given promise. This process must take place as a Job to ensure that the evaluation of the then method occurs after evaluation of any surrounding code has completed.

并且该 Jobs 还会调用一次 then 函数来 resolve Promise,这也就又生成了一次微任务。

这就是为什么会触发两次微任务的来源。

最后

文章到这里就完结了,大家有什么疑问都可以在评论区提出。

想了解更多编程学习,敬请关注php培训栏目!

위 내용은 진정으로 이해해야 할 약속의 상세 내용입니다. 자세한 내용은 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 옷 제거제

Video Face Swap

Video Face Swap

완전히 무료인 AI 얼굴 교환 도구를 사용하여 모든 비디오의 얼굴을 쉽게 바꾸세요!

뜨거운 도구

메모장++7.3.1

메모장++7.3.1

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

SublimeText3 중국어 버전

SublimeText3 중국어 버전

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

스튜디오 13.0.1 보내기

스튜디오 13.0.1 보내기

강력한 PHP 통합 개발 환경

드림위버 CS6

드림위버 CS6

시각적 웹 개발 도구

SublimeText3 Mac 버전

SublimeText3 Mac 버전

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

약속 지키기: 약속 이행의 장점과 단점 약속 지키기: 약속 이행의 장점과 단점 Feb 18, 2024 pm 08:06 PM

일상생활에서 우리는 약속과 이행 사이에서 종종 문제에 직면합니다. 개인적인 관계에서든 비즈니스 거래에서든 약속을 이행하는 것은 신뢰 구축의 핵심입니다. 그러나 헌신의 장단점은 종종 논란의 여지가 있습니다. 이 기사에서는 약속의 장단점을 살펴보고 약속을 지키는 방법에 대한 몇 가지 조언을 제공합니다. 약속된 혜택은 분명합니다. 첫째, 헌신은 신뢰를 구축합니다. 사람이 약속을 지키면 다른 사람들이 자신을 믿을 만한 사람이라고 믿게 만듭니다. 신뢰는 사람들 사이에 확립된 유대이며, 이는 사람들을 더 나은 사람으로 만들 수 있습니다.

Vue 애플리케이션에서 Uncaught (약속대로) TypeError가 발생하면 어떻게 해야 합니까? Vue 애플리케이션에서 Uncaught (약속대로) TypeError가 발생하면 어떻게 해야 합니까? Jun 25, 2023 pm 06:39 PM

Vue는 널리 사용되는 프런트엔드 프레임워크로, 애플리케이션을 개발할 때 다양한 오류와 문제가 자주 발생합니다. 그중 Uncaught(inpromise)TypeError가 일반적인 오류 유형입니다. 이번 글에서는 그 원인과 해결방법에 대해 알아보겠습니다. Uncaught(inpromise)TypeError란 무엇입니까? Uncaught(inpromise)TypeError 오류는 일반적으로 다음에서 발생합니다.

Promise.resolve()에 대해 자세히 알아보세요. Promise.resolve()에 대해 자세히 알아보세요. Feb 18, 2024 pm 07:13 PM

Promise.resolve()에 대한 자세한 설명에는 특정 코드 예제가 필요합니다. Promise는 비동기 작업을 처리하기 위한 JavaScript의 메커니즘입니다. 실제 개발에서는 순서대로 실행해야 하는 일부 비동기 작업을 처리해야 하는 경우가 종종 있으며, 이행된 Promise 객체를 반환하기 위해 Promise.resolve() 메서드가 사용됩니다. Promise.resolve()는 Promise 클래스의 정적 메서드입니다.

ES6 Promise의 원리와 활용 사례 분석 ES6 Promise의 원리와 활용 사례 분석 Aug 09, 2022 pm 03:49 PM

Promise 객체를 사용하여 일반 함수를 Promise를 반환하도록 변경하여 콜백 지옥 문제를 해결합니다. Promise의 성공 및 실패 호출 로직을 이해하고 유연하게 조정할 수 있습니다. 핵심지식을 이해하고, 먼저 활용하고, 천천히 지식을 통합하고 흡수하세요.

어떤 브라우저가 Promise를 지원하나요? 어떤 브라우저가 Promise를 지원하나요? Feb 19, 2024 pm 04:41 PM

브라우저 호환성: 어떤 브라우저가 Promise를 지원하나요? 웹 애플리케이션의 복잡성이 계속 증가함에 따라 개발자는 JavaScript의 비동기 프로그래밍 문제를 해결하기 위해 노력하고 있습니다. 과거에는 개발자가 비동기 작업을 처리하기 위해 콜백 함수를 자주 사용했지만 이로 인해 코드가 복잡하고 유지 관리가 어려워졌습니다. 이 문제를 해결하기 위해 ECMAScript6에서는 비동기 작업을 처리하는 보다 직관적이고 유연한 방법을 제공하는 Promise를 도입했습니다. Promise는 예외를 처리하는 데 사용되는 방법입니다.

Promise 객체란 무엇인가요? Promise 객체란 무엇인가요? Nov 01, 2023 am 10:05 AM

Promise 객체 상태는 다음과 같습니다. 1. 보류 중: 초기 상태, 성공 또는 실패 상태가 아닙니다. 2. 이행됨: 작업이 성공적으로 완료되었음을 의미합니다. 3. 거부됨: 작업이 실패했음을 의미합니다. Promise 객체가 완료되면 보류 상태에서 이행 또는 거부 상태로 변경되며 다시 변경할 수 없습니다. Promise 객체는 AJAX 요청 및 시간 제한 작업과 같은 비동기 작업을 처리하기 위해 JavaScript에서 널리 사용됩니다.

Promise 객체를 반환하는 PHP 함수의 장점은 무엇입니까? Promise 객체를 반환하는 PHP 함수의 장점은 무엇입니까? Apr 19, 2024 pm 05:03 PM

장점: 비동기 및 비차단, 기본 스레드를 차단하지 않음, 코드 가독성 및 내장 오류 처리 메커니즘 향상.

약속은 무슨 뜻인가요? 약속은 무슨 뜻인가요? Nov 02, 2023 pm 05:30 PM

Promise는 비동기 작업을 처리하기 위한 프로그래밍 패턴으로, 비동기 작업의 최종 완료 또는 실패를 나타내는 객체로, 비동기 코드를 더 잘 관리하고 구성할 수 있습니다. 더 읽기 쉽고 유지 관리가 쉽습니다. Promise 객체에는 보류, 이행, 거부의 세 가지 상태가 있습니다. Promise의 핵심 아이디어는 콜백 함수에서 비동기 작업을 분리하고 체인 호출을 통해 비동기 작업 간의 종속성을 표현하는 것입니다.

See all articles