详细解释浏览器是如何渲染页面的?

渲染流程概述

渲染的目标:将HTML文本转化为可以看到的像素点

当浏览器的网络线程收到 HTML 文档后,会产生一个渲染任务,并将其传递给渲染主线程的消息队列。在事件循环机制的作用下,渲染主线程取出消息队列中的渲染任务,开启渲染流程。

渲染流程概述

渲染流程的阶段(8个步骤):

在这里插入图片描述

  • HTML 解析
  • 样式计算
  • 布局
  • 分层
  • 绘制
  • 分块
  • 光栅化

每个阶段的输出将成为下一个阶段的输入,整个流程形成了一套高效的生产流水线。

1. 解析 HTML

第一步: 解析 HTML

在这里插入图片描述

  • HTML 解析: 浏览器会解析 HTML 文件,遇到 CSS 会解析 CSS,遇到 JS 会执行 JS
  • 外部资源的加载: 浏览器会启动一个预解析线程,提前下载外部的 CSSJS 文件。

关键点:

  • link 标签会在主线程解析到时,继续解析 HTML,不会阻塞。
    在这里插入图片描述
  • script 标签会暂停 HTML 解析,等待 JavaScript 下载并执行完成后,才能继续解析。
    HTML解析

解析完成后,浏览器得到 DOM 树CSSOM 树

2. 样式计算

第二步: 样式计算

主线程会遍历 DOM 树,计算每个节点的最终样式,称之为 Computed Style

这一过程存在转换操作,预设值(例如 red)转换成绝对值(例如 rgb(255, 0, 0)),单位(例如 em)转化为 px

此步骤后,我们得到了一棵包含样式的 DOM 树。

样式计算


3. 布局

第三步: 布局

布局阶段,浏览器会根据 DOM 树计算出每个节点的几何信息,如宽高、位置等。

布局树的特性(DOM树和Layout树不一定是一一对应的):

  • display: none 的节点不会出现在布局树中。
  • 伪元素 ::before 会出现在布局树中,尽管 DOM 树中没有。

完成布局后,浏览器会生成 布局树

布局


4. 分层

第四步: 分层

主线程会根据布局树,按照一套复杂的策略进行分层。

为什么要分层:

如果某一层发生变化,后续只处理这一层,从而提升性能。

影响分层的因素包括:滚动条、transformopacity 等,也可以通过will-change属性更大程度的影响分层结果。

分层


5. 绘制

第五步: 绘制

每一层的绘制指令集会被生成,用于描述图层如何呈现内容。

绘制

完成绘制后,主线程将绘制信息提交给 合成线程,剩下的工作由合成线程完成。

分块

6. 分块

第六步: 分块
在这里插入图片描述

合成线程将每一层分成多个小块,划分为更小的区域。

在这里插入图片描述

这一步会从线程池中取出多个线程来完成分块的工作。

7. 光栅化

第七步: 光栅化

合成线程将分块信息交给 GPU 进程,通过多个线程快速处理,优先处理靠近视口的区域。光栅化后的结果是每个块的位图。

光栅化
光栅化的过程中会启动GPU进程进行加速。
在这里插入图片描述

8. 画

最后一步:

合成线程生成 指引(quad)信息,描述每个位图如何绘制到屏幕的正确位置,并考虑旋转、缩放等变形。

变形发生在合成线程,与渲染主线程无关,正是因为如此,transform 样式非常高效。

最终,合成线程将指引信息提交给 GPU 进程,通过 GPU 硬件渲染生成屏幕上的图像。

画

总结

通过并行处理,浏览器能够在多个线程中同时完成渲染任务,保证页面能够快速响应用户交互,提高了整体性能。渲染流程的各个阶段和线程池的协作使得浏览器能够高效地绘制出用户所看到的页面,优化了体验和性能。

加速技巧:

  • will-change 属性可以提前告知浏览器需要优化的渲染层。
  • 使用 transformopacity 来避免页面重排,提高渲染效率。

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/bicheng/77519.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

java+postgresql+swagger-多表关联insert操作(九)

入参为json,然后根据需要对多张表进行操作: 入参格式: {"username": "车主01","usertel": "11111111111","useridtype": "2","useridcard": null,"proname&qu…

JavaSpring 中使用 Redis

创建项目 配置 Redis 服务地址 创建 Controller 类 由于当前只是些简单的测试代码,所以就不进行分层了,只创建一个 Controller 来实现 jedis 通过 jedis 对象里的各种方法来操作 Redis 此处通过 StringRedisTemplate 来操作 Redis 最原始提供的类是 Re…

AI文生图工具推荐

一、AI文生图技术实现原理 AI文生图(Text-to-Image)基于生成对抗网络(GAN)或扩散模型(Diffusion Model)实现,通过深度学习将文本描述转化为图像。其核心流程包括: 文本编码&#xf…

数据结构——快排和归并排序(非递归)

快速排序和归并排序一般都是用递归来实现的,但是掌握非递归也是很重要的,说不定在面试的时候面试官突然问你快排或者归并非递归实现,递归有时候并不好,在数据量非常大的时候效率就不好,但是使用非递归结果就不一样了&a…

【笔记】网络安全管理

计算机硬件中,运算器和控制器通常集成在一块芯片内,一般称为()。 数据库DB、数据库系统DBS、数据库管理系统DBMS,三者之间的关系是()。 OSI/RM体系结构中的网络层与TCP/IP体系结构中的()功能相同。 三级系统应按照等保2.0要求采用密码技术通信过程中数据的()。 …

.net core web api 数据验证(DataAnnotations)

目录 一、什么是 DataAnnotations? 二、扩展验证逻辑(自定义验证器) 一、什么是 DataAnnotations? DataAnnotations 是一组特性(Attributes),用于在模型类上定义验证规则。主要用于属性级别的…

小白从0学习网站搭建的关键事项和避坑指南

以下是针对小白从零学习网站搭建时需要注意的关键事项和避坑指南,帮助你高效学习、少走弯路: 一、学习路径注意事项 不要跳过基础 误区:直接学习框架(如 React、Laravel)而忽视 HTML/CSS/JS 基础。 正确做法&#xff…

深入剖析JavaScript内存泄漏:识别、定位与实战解决

在JavaScript的世界里,开发者通常不必像使用C那样手动管理内存的分配和释放,这得益于JavaScript引擎内置的垃圾回收(Garbage Collection, GC)机制。然而,这并不意味着我们可以完全忽视内存管理。“自动"不等于&qu…

2025-04-19 Python 强类型编程

文章目录 1 方法标注1.1 参数与返回值1.2 变参类型1.3 函数类型 2 数据类型2.1 内置类型2.2 复杂数据结构2.3 类别选择2.4 泛型 3 标注方式3.1 注释标注3.2 文件标注 4 特殊情形4.1 前置引用4.2 函数标注扩展4.3 协变与逆变4.4 dataclass 5 高级内容5.1 接口5.2 泛型的协变/逆变…

ETF价格相关性计算算法深度分析

1. 引言 在金融市场中,相关性就像是资产之间“跳舞”的默契程度。想象一下两位舞者(ETF),有时步伐一致,有时各跳各的。对于管理大规模资金的投资组合而言,准确理解ETF之间的“舞步同步性”对于风险管理、资…

上海人工智能实验室:LLM无监督自训练

📖标题:Genius: A Generalizable and Purely Unsupervised Self-Training Framework For Advanced Reasoning 🌐来源:arXiv, 2504.08672 🌟摘要 🔸推进LLM推理技能引起了广泛的兴趣。然而,当前…

【WPF】 在WebView2使用echart显示数据

文章目录 前言一、NuGet安装WebView2二、代码部分1.xaml中引入webview22.编写html3.在WebView2中加载html4.调用js方法为Echarts赋值 总结 前言 为了实现数据的三维效果,所以需要使用Echarts,但如何在WPF中使用Echarts呢? 一、NuGet安装WebV…

2025年3月 Python编程等级考试 2级真题试卷

2025年3月青少年软件编程Python等级考试(二级)真题试卷 题目总数:37 总分数:100 选择题 第 1 题 单选题 老师要求大家记住四大名著的作者,小明机智地想到了可以用字典进行记录,以下哪个选项的字典…

6. 话题通信 ---- 使用自定义msg,发布方和订阅方cpp,python文件编写

1)在功能包下新建msg目录&#xff0c;在msg目录下新建Person.msg,在Person.msg文件写入&#xff1a; string name uint16 age float64 height 2)修改配置文件 2.1) 功能包下package.xml文件修改 <build_depend>message_generation</build_depend><exec_depend…

多线程使用——线程安全、线程同步

一、线程安全 &#xff08;一&#xff09;什么是线程安全问题 多个线程&#xff0c;同时操作同一个共享资源的时候&#xff0c;可能会出现业务安全的问题。 &#xff08;二&#xff09;用程序摹拟线程安全问题 二、线程同步 &#xff08;一&#xff09;同步思想概述 解决线…

4. 话题通信 ---- 发布方和订阅方cpp文件编写

本节对应赵虚左ROS书籍的2.1.2 以10hz,发布消息和消息的订阅 1) 在功能包的src文件夹下&#xff0c;新建cpp文件&#xff0c;并且写入 #include "ros/ros.h" #include "std_msgs/String.h" int main(int argc, char *argv[]) {setlocale(LC_ALL,"&…

有哪些哲学流派适合创业二

好的&#xff0c;让我们更深入地探讨如何将‌哲学与数学‌深度融合&#xff0c;构建一套可落地的创业操作系统。以下从‌认知框架、决策引擎、执行算法‌三个维度展开&#xff0c;包含具体工具和黑箱拆解&#xff1a; ‌一、认知框架&#xff1a;用哲学重构商业本质‌ 1. ‌本体…

【后端】【python】Python 爬虫常用的框架解析

一、总结 Python 爬虫常用的框架主要分为 三类&#xff1a; 轻量级请求库&#xff1a;如 requests、httpx&#xff0c;用于快速发请求。解析与处理库&#xff1a;如 BeautifulSoup、lxml、pyquery。爬虫框架系统&#xff1a;如 Scrapy、pyspider、Selenium、Playwright 等&am…

力扣-hot100(无重复字符的最长子串)

3. 无重复字符的最长子串 中等 给定一个字符串 s &#xff0c;请你找出其中不含有重复字符的 最长 子串 的长度。 示例 1: 输入: s "abcabcbb" 输出: 3 解释: 因为无重复字符的最长子串是 "abc"&#xff0c;所以其长度为 3。暴力直观解法一&#xff1…

六边形棋盘格(Hexagonal Grids)的坐标

1. 二位坐标转六边形棋盘的方式 1-1这是“波动式”的 这种就是把【方格子坐标】“左右各错开半个格子”做到的 具体来说有如下几种情况 具体到庙算平台上&#xff0c;是很巧妙的用一个4位整数&#xff0c;前两位为x、后两位为y来进行表示 附上计算距离的代码 def get_hex_di…