为什么 DOMContentLoaded 处理程序可以阻止首次绘制?

Rom*_*man 9 javascript dom domcontentloaded

有一个类似的问题没有解决。

我遇到过侦听 DOMContentLoaded 的处理程序可以阻止首次绘制的情况。有时会阻塞,有时不会

我尝试了很多次 cmd + R 来查看它。这种行为有什么解释吗?

我还录制了一段视频来展示这一点:https : //www.youtube.com/watch?v=EDZQ1nLCK2w&feature=youtu.be

  1. 当你在重新加载后看到一个空白页面时,这意味着 DOMContentLoaded 阻止了第一次绘制
  2. 当您看到文本“Some text”然后重新加载后出现空白页面时,这意味着 DOMContentLoaded 没有阻止第一次绘制
window.addEventListener('DOMContentLoaded', () => {
    let i = 0;
    while (i++ < 1000000000) {
        continue;
    }
    document.getElementById('el').remove();
});
Run Code Online (Sandbox Code Playgroud)
<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>
<body>
    <p id="el">Some text</p>
</body>
</html>
Run Code Online (Sandbox Code Playgroud)

Rom*_*man 0

我想到了。Blink渲染引擎存在同步调度 DOMContentLoaded 事件的错误

// #blink/renderer/core/dom/document.cc
// #blink::Document::FinishedParsing

// FIXME: DOMContentLoaded is dispatched synchronously, but this should be
// dispatched in a queued task, see https://crbug.com/425790
if (document_timing_.DomContentLoadedEventStart().is_null())
  document_timing_.MarkDomContentLoadedEventStart();
DispatchEvent(*Event::CreateBubble(event_type_names::kDOMContentLoaded));
if (document_timing_.DomContentLoadedEventEnd().is_null())
  document_timing_.MarkDomContentLoadedEventEnd();
SetParsingState(kFinishedParsing);
Run Code Online (Sandbox Code Playgroud)

** 为什么有时会异步发送我不知道。现在,在我这边,它总是同步的(可能在提出问题时还有一个错误)