参考答案:
浏览器内有多个进程,其中渲染进程被称为浏览器内核,负责页面渲染和执行 JS 脚本等。渲染进程负责浏览器的解析和渲染,内部有 JS 引擎线程、 GUI 渲染线程、事件循环管理线程、定时器线程、HTTP 线程。
JS 引擎线程负责执行 JS 脚本,GUI 渲染线程负责页面的解析和渲染,两者是互斥的,也就是执行 JS 的时候页面是停止解析和渲染的。这是因为如果在页面渲染的同时 JS 引擎修改了页面元素,比如清空页面,会造成后续页面渲染的不必要和错误。而由于 JS 经常要操作 DOM ,就要涉及 JS 引擎线程和 GUI 渲染线程的通信,而线程间通信代价是非常昂贵的,这也是造成 JS 操作 DOM 效率不高的原因。
浏览器的 HTML/CSS 的解析和渲染都属于 GUI渲染线程,所以和 JS 引擎线程是互斥、阻塞的。下面从代码实际运行的角度分析浏览器解析和渲染的顺序,以及互相间的阻塞关系。
1<button class="btn btn-primary">test1</button> 2<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@4.4.1/dist/css/bootstrap.min.css"> 3<div>test2</div>
1<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@4.4.1/dist/css/bootstrap.min.css"> 2<script> 3 alert('ok') 4</script>
1<script> 2 alert('ok') 3</script> 4<link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@4.4.1/dist/css/bootstrap.min.css">
所以在需要提前执行不操作 dom 元素的 js 时,不妨把 js 放到 css 文件之前。
js 文件的下载和解析会阻塞 GUI 渲染进程,也就是会阻塞 DOM 和 CSS 的解析和渲染。
js 文件没下载并解析完成之前,后续的 HTML 和 CSS 无法解析:
1<script src="https://code.jquery.com/jquery-3.4.1.js"></script> 2<div>test</div>
1<div>test</div> 2<script src="https://code.jquery.com/jquery-3.4.1.js"></script>
1 <div>test1</div> 2 <script src="https://code.jquery.com/jquery-3.4.1.js"></script> 3 <div>test2</div>
1<body> 2 <script> 3 document.body.remove() 4 </script> 5 <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bootstrap@4.4.1/dist/css/bootstrap.min.css"> 6 <script src="https://code.jquery.com/jquery-3.4.1.js"></script> 7</body>
最近更新时间:2024-08-10