浏览器渲染原理(面试重点)

一、浏览器是如何渲染页面的

常见的简洁答案:
  1. 浏览器内核拿到内容后,渲染流程大致如下:
  2. 解析HTML,构建Dom树;
  3. 解析CSS,构建Render树;(将CSS代码解析成树形的数据结构,与Dom树结合形成Render树)
  4. 布局Render树(Layout/reflow),负责各元素尺寸、位置的计算;
  5. 绘制Render树(painting),绘制页面像素信息;
  6. 浏览器将各层的信息发送给GPU,GPU会将各层合成(composite),显示在屏幕上
牛逼的震惊面试官的答案:
  1. 当浏览器的网络线程收到HTML文档后,会产生一个渲染任务,并将其传递给渲染主线程的消息队列。在事件循环机制的作用下,渲染主线程取出消息队列中的渲染任务,开启渲染流程。
  2. 整个渲染流程分为多个阶段,分别是:HTML解析、样式计算、布局、分层、绘制、分块、光栅化、画。每个阶段都有明确的输入和输出,上一个阶段的输出会成为下一个阶段的输入,这样整个渲染流程就形成了一套组织严密的生产流水线。
  3. 渲染的第一步是解析HTML:
    1. 解析过程中遇到CSS就解析CSS,遇到JS就解析JS。为了提高解析效率,浏览器在开始解析前,会启动一个预解析的线程,率先下载HTML中的外部CSS文件和外部的JS文件
    2. 如果主线程解析到link位置,此时外部的CSS文件还没有下载解析好,主线程不会等待,继续解析后续的HTML。这是因为下载和解析CSS的工作是在预解析线程中进行的。这就是CSS不会阻塞HTML解析的根本原因。
    3. 如果主线程解析到script位置,会停止解析HTML,转而等待JS文件下载好,并将全局代码解析执行完成后,才能继续解析HTML。这是因为JS代码执行的过程可能会修改当前的DOM树,所以DOM树的生成必须暂停。这就是JS会阻塞HTML解析的根本原因。
    4. 第一步完成后,会得到DOM树和CSSOM树,浏览器的默认样式、内部样式、外部样式、行内样式均会包含在CSSOM树中。
  4. 渲染的下一步是样式计算:
    1. 主线程会遍历得到DOM树,依次为树中的每个节点计算出它最终的样式,称之为Computed Style。
    2. 在这一过程中,很多预设值会变成绝对值,比如red会变成rgb(255,0,0);相对单位会变成绝对单位,比如em会变成px。
    3. 这一步完成后,会得到一棵带有样式的DOM树
  5. 接下来是布局,布局完成后会得到布局树:
    1. 布局阶段会依次遍历DOM树的每一个节点,计算每个节点的几何信息。例如节点的宽高、相对包含块的位置。
    2. 大部分时候,DOM树和布局树并非一一对应。比如display: none;的节点没有几何信息,因此不会生成到几何树;又比如使用了伪元素选择器,虽然DOM树中不存在这些伪元素节点,但他们拥有几何信息,所以会生成到布局树中。还有匿名行盒、匿名块盒等等都会导致DOM树和布局树无法一一对应。
  6. 下一步是分层:
    1. 主线程会使用一套复杂的策略对整个布局树中进行分层
    2. 分层的好处在于,将来某一个层改变后、仅会对该层进行后续处理,从而提升效率。
    3. 滚动条、折叠上下文、transform、opacity等样式都会或多或少的影响分层效果,也可以通过will-change属性更大程度的影响分层效果。
  7. 再下一步是绘制:
    1. 主线程会为每个层单独绘制指令集,用于描述这一层的内容该如何画出来
    2. 完成绘制后,主线程将每个图层的绘制信息提交给合成线程,剩余工作将由合成线程完成。
    3. 合成线程首先对每个图层进行分块 、将其划分为更小的区域。它会从线程池中拿取多个线程来完成分块工作。
  8. 分块完成后进入光栅化阶段:
    1. 合成线程会将块信息交给GPU进程,以极高的速度完成光栅化
    2. GPU进程会开启多个线程来完成光栅化,并且有限处理靠近视口区域的块。
    3. 光栅化的结果就是一块一块的位图。
  9. 最后一个阶段,就是画了:
    1. 合成线程拿到每个层、每个块的位图后,生成一个个指引信息
    2. 指引会标识出每个位图应该画到屏幕的哪个位置,以及会考虑到旋转、缩放等变形。
    3. 变形发生在合成线程,与渲染主线程无关,这就是transform效率高的本质原因。
    4. 合成线程会把指引信息交给GPU进程,由GPU进程产生系统调用,提交给GPU硬件,完成最终屏幕的生成

二、什么是reflow(回流)?

reflow的本质就是重新计算layout树当进行了会影响布局树的操作后,需要重新计算布局树,会引发layout

为了避免连续的多次操作导致布局树反复计算,浏览器会合并这些操作,当JS代码全部完成后再进行统一计算。所以,改动属性造成的reflow是异步完成的。

也同样因为如此,当JS获取到布局属性时,就可能造成无法获取到最新的布局消息。

浏览器在反复权衡下,最终决定获取属性立即reflow。

三、什么时repaint(重绘)?

repaint的本质就是重新根据分层信息计算了绘制指令。当改动了可见样式后就需要重新计算会引发repaint

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

四、为什么transform的效率高?

因为transform既不会影响布局也不会影响绘制指令,它影响的只是渲染流程的最后一个“画”阶段,由于“画”阶段在合成线程中,所以transform的变化几乎不会影响渲染主线程。反之,渲染主线程无论如何忙碌,也不会影响transform的变化。

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

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

相关文章

使用nsight system 分析python进程

有时候我们的推理引擎是通过python脚本调用的,比如: python脚本调用TensorRT engine进行推理。 如果我们想用nsight system 分析性能,该怎么搞呢? 方法如下: 首先直接nsys profile 后面跟要执行的python命令就行 $…

一个vue3的tree组件

https://download.csdn.net/download/weixin_41012767/88709466

lldb in android studio

命令 n 单步s 进入函数, step intofini 函数返回, step outc 继续执行bt 显示调用栈, 标记*的行, 为当前帧(函数)frame info 显示栈当前所在的当前帧frame select [数字] 在当前栈中选择某个帧var 显示所有局部变量dis 显示汇编, 箭头位置为执行位置ni 汇编单步si 汇编进入函数…

c# vb.net检测字符串是否匹配一组相似度数组input Like

VB.NET 检测字符串是否符合一个数组中的多个like条件,有没有最简单的函数? 在VB.NET中,可以使用Enumerable.Any方法结合String.Like方法来检测一个字符串是否符合一个数组中的多个LIKE条件。Enumerable.Any方法用于确定序列中的任何元素是否…

java SSM社区文化服务管理系统myeclipse开发mysql数据库springMVC模式java编程计算机网页设计

一、源码特点 java SSM社区文化服务管理系统是一套完善的web设计系统(系统采用SSM框架进行设计开发,springspringMVCmybatis),对理解JSP java编程开发语言有帮助,系统具有完整的 源代码和数据库,系统主…

Mac环境下Parallels Desktop 19的安装和使用

为了后续构建漏洞靶场和渗透测试环境,我们需要提前准备好几套与宿主机隔离的工作环境(Windows、Linux等),在Mac上最常用的就是Paralles Desktop(PD)工具了,当前最新版本为19。接下来介绍如何安装…

Http与Tcp协议的原理以及应用

OSI七层模型和相关协议 七层模型从上到下如下所示: 应用层:负责应用之间的通信,处理请求和响应的具体格式表示层:对于数据格式进行处理会话层:负责建立和断开通信连接,传输层:负责建立端口之间…

python开发案例教程-清华大学出版社(张基温)答案(4.3)

练习 4.1 1. 判断题 判断下列描述的对错。 (1)子类是父类的子集。 ( ) (2)父类中非私密的方法能够被子类覆盖。 ( ) (3)子类能够…

java distinct 无法为泛型去重

针对distinct无法为List<User>之类的去重。 所以这里简单说一下方案。 本质上是重写对象的equals方法。 但是这里有一个偷懒的方式。 就是直接使用Data (来自Lombok的注解) 因为Data 一样会重写equals 和 hashcode方法。 所以&#xff0c;直接使用Data来代替get;se…

8086控制电机旋转,设了三个指示灯不亮咋回事,是电压不够?还是保护电阻太大了?

8086控制电机旋转&#xff0c;设了三个指示灯不亮咋回事&#xff0c;是电压不够&#xff1f;还是保护电阻太大了&#xff1f; 在控制电机旋转时&#xff0c;如果设定了三个指示灯但它们不亮&#xff0c;可能有几个原因导致&#xff1a; 1. 电压不够&#xff1a;确保电机和指示灯…

Rust 字符串 初步了解

rust 的字符串 。字符串不是复合类型&#xff0c; String 和 &str &#xff1a; String 具有所有权&#xff0c;是存储在堆上的。&str 没有所有权&#xff0c;是对 String 的引用。字符串字面量也是 &str 类型&#xff0c;存储在栈上。 切片&#xff08;slice&a…

java基于ssm的房源管理系统+vue论文

目 录 目 录 I 摘 要 III ABSTRACT IV 1 绪论 1 1.1 课题背景 1 1.2 研究现状 1 1.3 研究内容 2 2 系统开发环境 3 2.1 vue技术 3 2.2 JAVA技术 3 2.3 MYSQL数据库 3 2.4 B/S结构 4 2.5 SSM框架技术 4 3 系统分析 5 3.1 可行性分析 5 3.1.1 技术可行性 5 3.1.2 操作可行性 5 3…

【python爬虫开发实战 情感分析】利用爬虫爬取城市评论并对其进行情感分析

&#x1f680;个人主页&#xff1a;为梦而生~ 关注我一起学习吧&#xff01; &#x1f4a1;专栏&#xff1a; python网络爬虫从基础到实战 带你学习爬虫从基础到实战 深度学习带你感受AI的魅力 &#x1f4a1;往期推荐&#xff1a; ⭐️前面比较重要的基础内容&#xff1a; 【Py…

Linux vi/vim 教程

文章目录 【 1. vi/vim 的三种模式 】1.1 命令模式1.2 输入模式1.3 底线命令模式 【 2. 实例 】【 3. vim 的其他命令 】 所有的 Unix Like 系统都会内建 vi 文本编辑器&#xff0c;其他的文本编辑器则不一定会存在。目前我们使用比较多的是 vim 编辑器。vim 从 vi 发展出来&am…

STM32G030C8T6:USART串口通信(中断)

本专栏记录STM32开发各个功能的详细过程&#xff0c;方便自己后续查看&#xff0c;当然也供正在入门STM32单片机的兄弟们参考&#xff1b; 本小节的目标是&#xff0c;系统主频64 MHZ,采用高速外部晶振&#xff0c;通过芯片PB6,PB7 的USART1 口&#xff0c;实现串口通信。 原理…

Docker nginx容器代理播放m3u8视频文件(HLS)

文章目录 Docker Nginx容器代理播放M3U8文件教程获取Nginx Docker镜像设置Nginx配置文件用 ffmpeg 将 MP4 文件转换成 m3u8 文件运行Docker容器测试M3U8流其他问题我用vlc都能播放http://192.168.121.50/forest4kTest.m3u8和http://192.168.121.50/forest4kTest.mp4&#xff0c…

用opencv的DNN模块做Yolov5目标检测(纯干货,源码已上传Github)

最近在微信公众号里看到多篇讲解yolov5在openvino部署做目标检测文章&#xff0c;但是没看到过用opencv的dnn模块做yolov5目标检测的。于是&#xff0c;我就想着编写一套用opencv的dnn模块做yolov5目标检测的程序。在编写这套程序时&#xff0c;遇到的bug和解决办法&#xff0c…

机器学习:手撕 AlphaGo(二)

计算机下围棋的问题描述请见上篇&#xff1a;机器学习&#xff1a;手撕 AlphaGo&#xff08;一&#xff09;-CSDN博客 3. MCTS 算法介绍 MCTS&#xff08;Monte Carlo Tree Search&#xff09; 算法的中文名称叫做蒙特卡洛树搜 索。第一次接触这个算法时&#xff0c;便惊叹于它…

Java 说一下 synchronized 底层实现原理?

Java 说一下 synchronized 底层实现原理&#xff1f; synchronized 是 Java 中用于实现同步的关键字&#xff0c;它保证了多个线程对共享资源的互斥访问。底层实现涉及到对象头的 Mark Word 和锁升级过程。 synchronized 可以用于方法上或代码块上&#xff0c;分别对应于方法…

Ubuntu 本地部署 ChatGPT-Next-Web

Ubuntu 本地部署 ChatGPT-Next-Web 文章目录 Ubuntu 本地部署 ChatGPT-Next-Web ChatGPT-Next-Web 项目地址&#xff1a;https://github.com/ChatGPTNextWeb/ChatGPT-Next-Web 本文主要演示如何在 Ubuntu 本地&#xff08;默认是端口 3000&#xff09;部署 ChatGPT-Next-Web&am…