问小白 wenxiaobai
资讯
历史
科技
环境与自然
成长
游戏
财经
文学与艺术
美食
健康
家居
文化
情感
汽车
三农
军事
旅行
运动
教育
生活
星座命理

浏览器渲染揭秘:从加载到显示的全过程

创作时间:
作者:
@小白创作中心

浏览器渲染揭秘:从加载到显示的全过程

引用
CSDN
1.
https://blog.csdn.net/RenGJ010617/article/details/140212872

浏览器渲染是前端开发中一个非常重要且复杂的主题。从页面加载到最终显示,浏览器需要经过多个步骤的处理。本文将详细解析浏览器渲染的全过程,包括HTML解析、样式计算、布局、分层、绘制、分块、光栅化和最终绘制等关键步骤,并解释reflow和repaint等常见性能问题,帮助读者深入理解浏览器的工作原理。

浏览器工作原理与流程

一、渲染开始时间点

用户访问页面的时候,浏览器网络线程进行网络通信获取HTML代码,然后进入渲染主线程的消息队列进行包装,得到渲染任务后按需交由渲染主线程进行渲染。

二、渲染主线程的渲染流程

2.1、渲染流程总览

总流程如下图所示:

2.2、渲染具体步骤

①解析html-Parse HTML

该步骤主要生成DOM树和CSSOM树。html代码解析后生成DOM树,css代码解析后生成CSSOM树。

在解析过程中,为了提⾼解析效率,浏览器会启动⼀个预解析器率先下载和解析 CSS。渲染主线程遇到 JS 时必须暂停⼀切⾏为,等待下载执⾏完后才能继续,预解析线程可以分担⼀点下载 JS 的任务,这里主要是因为JS可能会改变DOM树, 所以需要先解析,但如果声明是异步加载,那就不会暂停。

②样式计算-Recalculate Style

样式计算的目的是将HTML解析获得的DOM树和CSSOM树对应起来,只有DOM与CSSOM对应起来才能进行布局,为后面生成布局树奠定基础。

有一种说法将DOM树与CSSOM树绑定后的树称为“渲染树”。笔者个人认为在布局树还没生成之前,叫它为渲染树有些过早了。

③布局-Layout

根据计算好的样式来生成布局树。布局树包含了之后渲染后呈现给我们的所有信息。

这里有一个误区,就是认为布局情况依赖DOM元素,所以布局树和DOM树应该是一一对应的。这句话是错误的,只能说布局树和可见内容是一一对应的,但是和DOM树并不一定一一对应。反例包括:

1、display值为none的元素,只是不可见,但是依然会添加到DOM树中,而布局树中不会添加该元素。

2、before伪类添加的元素,由于不算做一个单独的元素,所以不会添加到DOM树中,但是会在布局中显示,所以会添加到布局树中。

3、浏览器在渲染过程中为了保持布局的连贯性和符合CSS规范会自动创建一些布局盒子:匿名行盒与匿名块盒,它们只会在布局树中添加,而不会影响DOM树。

④分层-Layer

在现代浏览器中,为了优化渲染性能,渲染树会被分解成多个层(Layers)。每个层可以独立于其他层进行渲染和合成,这有助于减少绘制和合成的工作量。

跟堆叠上下文有关的属性,会影响分层,比如z-index,opacity、transform、filter,或者元素被设置为will-change,以及部分动画和过度效果可能也会被分为新的层。

相关拓展

opacity属性与堆叠上下文有关,因为当元素的opacity值小于1时,它将创建一个内部的堆叠上下文。这意味着,即使元素的z-index值较低,它的不透明部分仍然可以覆盖在其后面的元素的不透明部分。这可能导致一些不可预见的层叠效果,因为元素的不透明度会影响其在堆叠上下文中的行为。

例如,假设有两个元素A和B,A在B的上方,A的z-index值较高,但A的opacity为0.5,B的opacity为1。在这种情况下,A的不透明部分将覆盖B,但A的半透明部分将允许B的内容显示出来。这是因为A的半透明部分创建了一个内部堆叠上下文,而B的内容在这个内部堆叠上下文中显示。

堆叠上下文的层级规则如下:

  • 根元素(通常是HTML元素)形成一个堆叠上下文。
  • 定位元素(position属性为relative、absolute或fixed)可以形成新的堆叠上下文。
  • 元素的z-index属性可以指定其在堆叠上下文中的层级。
  • 某些CSS属性和值,如opacity、transform、filter等,可以创建新的堆叠上下文。

⑤绘制-Paint

为分层结果的每⼀层⽣成如何绘制的指令,并不是真的绘制。

到这里渲染主线程的工作就结束了,剩下的任务交给其他线程完成。

⑥分块-Tiling

这一步会将每⼀层分为多个小的区域。分块的目的是对分层结果进行进一步细分,通过只渲染用户可以看到的部分(即视口中的部分),浏览器可以更快地完成绘制工作。当用户滚动页面时,浏览器可以丢弃不在视口中的块,并重新绘制新进入视口的块。

以此来减少内存使用,提高渲染效率,优化重绘制,改善滚动性能。

⑦光栅化-Raster

光栅化将分块后的每个块都绘制成位图,优先处理靠近视口的块,以此提高渲染效率。这一过程需要GPU加速。

有些说法将⑤⑥⑦三个部分统一为一个步骤“绘制”,其实是一个道理,只是颗粒度不同。

⑧画-Draw(合成)

合成线程计算出每个位图在屏幕上的位置,交给GPU进行最终呈现。

⑨完整过程分工示意图

完整过程的分工情况如图:

三、衍生问题

3.1、什么是reflow?

reflow 的本质就是重新计算 layout 树。

用js修改样式,即修改cssom树,如果修改几何信息,就会修改dom。修改了几何信息就会导致reflow(改变可见dom的结构、修改宽高等)

reflow本身是异步请求,但是如果js中需要读取dom信息如dom.clientWidth,就会立即reflow来保证读取的信息是修改后的dom信息。

3.2、什么是repaint

repaint 的本质就是重新根据分层信息计算了绘制指令。

当改动了可见样式后,就需要重新计算,会引发 repaint。

由于元素的布局信息也属于可见样式,所以 reflow 一定会引起 repaint。

3.3、为什么transform效率高

因为 transform 既不会影响布局也不会影响绘制指令,它影响的只是渲染流程的最后一个「draw」阶段

由于 draw 阶段在合成线程中,所以 transform 的变化几乎不会影响渲染主线程。反之,渲染主线程无论如何忙碌,也不会影响 transform 的变化。

四、总结与相关资源

了解浏览器工作原理与流程,能有效帮助前端开发与性能优化。

© 2023 北京元石科技有限公司 ◎ 京公网安备 11010802042949号