我的编程空间,编程开发者的网络收藏夹
学习永远不晚

深入了解JavaScript阻塞渲染

短信预约 -IT技能 免费直播动态提醒
省份

北京

  • 北京
  • 上海
  • 天津
  • 重庆
  • 河北
  • 山东
  • 辽宁
  • 黑龙江
  • 吉林
  • 甘肃
  • 青海
  • 河南
  • 江苏
  • 湖北
  • 湖南
  • 江西
  • 浙江
  • 广东
  • 云南
  • 福建
  • 海南
  • 山西
  • 四川
  • 陕西
  • 贵州
  • 安徽
  • 广西
  • 内蒙
  • 西藏
  • 新疆
  • 宁夏
  • 兵团
手机号立即预约

请填写图片验证码后获取短信验证码

看不清楚,换张图片

免费获取短信验证码

深入了解JavaScript阻塞渲染

前言:

在中文社区,这么多年一直流传一个说法:JS线程负责执行JSGUI渲染线程负责渲染,这两者是互斥的,所以JS执行时会阻塞渲染。但随着Dev Tools使用的增多,逐渐开始怀疑以上说法。本文会以实际案例来解释为什么JS阻塞渲染。

到底几个线程

在讲解JS线程与GUI线程互斥的文章中,通常会列出渲染进程包含的线程,比如:

  • GUI渲染线程
  • JS引擎线程
  • 事件触发线程
  • 定时触发器线程
  • HTTP请求线程

但是,我们以百度的搜索页举例,打开Performance面板开启录制:

上图录制结果中:

  • Chrome_ChildIOThread对应IO线程的任务记录,用户输入、网络、设备相关事件都与他相关
  • Raster记录光栅化线程池任务、GPU记录GPU合成位图的任务、Compositor记录合成线程的任务执行,以上三者都与浏览器渲染相关
  • Main记录渲染进程的主线程中的任务

从这个角度看,浏览器实际的线程情况与那些GUI线程相关的文章描述的并不相同。

主线程的任务

接下来,让我们进入Main。红线框内长短不一的灰色块,就是主线程中执行的任务。

注意看红框内的绿色块FP,代表First Paint(首次绘制):

那么在首次绘制前都要执行什么任务呢?可以看到主要有3个Task(任务):

第一个任务是请求HTML数据:

Parse HTML

当请求回HTML字节流后,开始第二个任务,将HTML字节流解析为DOM,这个任务的名字就是图中的蓝色块Parse HTML

注意其中有些执行时长不一的Evaluate Script,这些是解析DOM树过程中遇到的JS代码。

DOM树中可以看到这些阻塞DOM树生成的JS脚本:

他们的存在显著拉长了Parse HTML的用时。

Recaculate Style

解析完DOM树(蓝色Parse HTML)后,下一个任务是紫色Recaculate Style

他负责将HTML中的CSS样式(外联、内联)输出为styleSheetsstyleSheets有两个作用:

  • 可以与DOM树结合为页面带来样式
  • JS可以操作styleSheets改变页面样式

我们可以从控制台打印document.styleSheets直观感受他的存在:

Layout

有了DOM树与styleSheets,接下来需要为视图中可见部分生成一棵树(比如display: none部分就不需要在这棵树中显示)。

这个任务是紫色Layout

Update Layer Tree

用户看到的页面实际是由多层页面重叠后的结果,开发者可以用很多手段(比如z-index)改变某部分的层级。

比如滚动条就会形成自己独立的层级:

既然是多层结构,那么就需要更新每层的信息,这个任务是紫色的Update Layer Tree

Paint

我们可以发现,在FP之前,Update Layer Tree之后只剩下Paint这一任务了:

从字面意义讲,这就是绘制么?并不是。

Paint的任务是整理每一层页面的绘制信息,构成绘制列表,这些数据会交给合成线程负责后续绘制操作。

可以发现,具体的绘制操作是交由合成线程完成,他与JS所在线程(主线程)并不是互斥的。

JS为啥阻塞渲染

我们现在知道,JS执行与Paint任务都发生在主线程。

渲染被阻塞的原因很明显:因为Paint任务没有及时执行,即绘制列表没有及时提交给合成线程。

之所以没有及时执行,可能是因为JS执行时间过长,导致这一帧没有时间执行Paint

比如,我们打开B站,记录下主线程的任务。

可以看到,有个JS执行时长达到231.88ms,超过了一帧的时间,在此期间主线程就没时间执行Paint了:

总结

JS之所以阻塞渲染,是因为JS执行与渲染相关任务都在争夺主线程有限的资源。当JS执行时间过长,渲染相关任务就没时间执行了。

到此这篇关于深入了解JavaScript阻塞渲染的文章就介绍到这了,更多相关JS阻塞渲染 内容请搜索编程网以前的文章或继续浏览下面的相关文章希望大家以后多多支持编程网!

免责声明:

① 本站未注明“稿件来源”的信息均来自网络整理。其文字、图片和音视频稿件的所属权归原作者所有。本站收集整理出于非商业性的教育和科研之目的,并不意味着本站赞同其观点或证实其内容的真实性。仅作为临时的测试数据,供内部测试之用。本站并未授权任何人以任何方式主动获取本站任何信息。

② 本站未注明“稿件来源”的临时测试数据将在测试完成后最终做删除处理。有问题或投稿请发送至: 邮箱/279061341@qq.com QQ/279061341

深入了解JavaScript阻塞渲染

下载Word文档到电脑,方便收藏和打印~

下载Word文档

猜你喜欢

Android 渲染机制深入理解

基础知识 CPU: 中央处理器,它集成了运算,缓冲,控制等单元,包括绘图功能.CPU将对象处理为多维图形,纹理(Bitmaps、Drawables等都是一起打包到统一的纹理)。 GPU:一个类似于CPU的专门用来处理Graphics的处理器
2022-06-06

Flutter渲染原理深入解析

众所周知 Flutter是由Google推出的开源的高性能跨平台框架,一个2D渲染引擎。在Flutter中,Widget是Flutter用户界面的基本构成单元,可以说一切皆Widget。下面来看下Flutter框架的整体结构组成
2023-05-15

深入解析Canvas的渲染模式

Canvas的renderMode详解,需要具体代码示例在Unity中,Canvas是实现2D UI最基础和关键的组件。Canvas在渲染过程中有两种不同的模式:Screen Space和World Space。这些渲染模式在特定的情况下
深入解析Canvas的渲染模式
2024-01-17

深入了解CSS布局重新计算和渲染的机制

CSS回流(reflow)和重绘(repaint)是网页性能优化中非常重要的概念。在开发网页时,了解这两个概念的工作原理,可以帮助我们提高网页的响应速度和用户体验。本文将深入探讨CSS回流和重绘的机制,并提供具体的代码示例。一、CSS回流
深入了解CSS布局重新计算和渲染的机制
2024-01-26

深入了解JavaScript发布订阅模式

JavaScript 发布订阅模式(Publish/Subscribe Pattern)是一种常用的设计模式,发布订阅模式的核心思想是解耦事件的发生和事件的处理,本文将介绍 JavaScript 发布订阅模式的基本原理、应用场景以及各场景的代码示例,需要的朋友可以参考下
2023-05-19

深入了解JavaScript中的函数柯里化

JavaScript函数柯里化是一种将接受多个参数的函数转换为一系列接受单个参数的函数的技术。本文将通过简单的示例为大家详细讲讲函数柯里化的相关应用,需要的可以参考一下
2023-05-16

深入了解JavaScript中的垃圾回收机制

JavaScript中的垃圾回收机制负责自动管理内存,回收不再使用的对象所占用的内存空间。本文主要介绍了JS中垃圾回收机制的相关知识,需要的可以参考一下
2023-05-15

JavaScript 错误对象:深入了解类型及其含义

JavaScript 错误对象:深入理解类型及其含义
JavaScript 错误对象:深入了解类型及其含义
2024-03-05

编程热搜

目录