四、浏览器渲染1.DOM、2.CSSOM、3.CSSComputed、4.Layout-布局、5.Layer-分层、6.Pain-绘制、7.Tiling-分块、8.Raster-光栅、9.Draw-画

知识点:
1、为什么不能先执行 js文件??
我们不能先执行JS文件,必须等到CSSOM构建完成了才能执行JS文件,因为前面已经说过渲染树是需要DOM和CSSOM构建完成了以后才能构建,而且JS是可以操控CSS样式的,所以这一步就是解析CSS文件并且构建CSSOM
2、DOM和CSSOM不同之处??
CSSOME的构建就是渲染中一个重要的阻塞因素,其实DOM也是会阻塞渲染过程,毕竟没有DOM,网页的框架都建造不起来,但是DOM有一点好处,就是可以部分解析,而CSSOM不能部分解析
3、 为什么DOM可以部分解析,CSSOM却不能?
比如我们给body设置了字体为32px,然后我们又给body里面的div设置了字体为16px,如果CSSOM只解析了body,后面的div没有解析或者延迟解析,那就会乱套了,所以CSSOM不能进行部分解析

在这里插入图片描述

4、渲染HTML+css+javascript

在这里插入图片描述
不过在解析CSS文件并且构建CSSOM的时候,浏览器依旧可以去下载并且解析JS文件,等CSSOM构建完成以后就可以执行JS里的内容了,不过大家有没有发现HTML的解析刚刚就停止了,原因可能很多同学都知道了,因为JS是会阻塞HTML解析的,虽然看起来JS并没有直接阻塞渲染的过程,但是有间接的影响因为JS既可以操作DOM,又可以操作CSSOM如果不等JS下载解析执行完以后再构建DOM,那有可能会导致网页的有些内容出现了又消失,所以在解析HTML的时候,不管是行内JS代码还是外部JS文件,都会让HTML的解析停止下来,虽然DOM是可以部分解析的,但是对于这个网页来说就相当于阻塞了第一次的渲染,JS执行完成之前什么内容都没有,JS执行完之后什么都正常了,也就是形成渲染树(构建DOM后),进行布局,最后绘制
在这里插入图片描述

❓常⻅问题:

DOM 树,执行一行,构建一行,最终js的执行(也需要看js的位置),需要等CSSDOM都执行完

HTML 解析过程中遇到 CSS / JS 代码怎么办?

在这里插入图片描述

在这里插入图片描述
1、渲染过程中,如果遇到 script标签 就停止渲染,执行 JS 代码。因为浏览器有GUI渲染线程与JS引擎线程,为了防止渲染出现不可预期的结果,这两个线程是互斥的关系。
2、JavaScript的加载、解析与执行会阻塞DOM的构建,也就是说,在构建DOM时,HTML解析器若遇到了JavaScript,那么它会暂停构建DOM,将控制权移交给JavaScript引擎,等JavaScript引擎运行完毕,浏览器再从中断的地方恢复DOM构建。
3、也就是说,如果你想首屏渲染的越快,就越不应该在首屏就加载 JS 文件,这也是都建议将 script 标签放在 body 标签底部的原因。当然在当下,并不是说 script 标签必须放在底部,因为你可以给 script 标签添加 defer 或者 async 属性(下文会介绍这两者的区别)。

❓疑问?JS文件不只是阻塞DOM的构建,它会导致CSSOM也阻塞DOM的构建。

原本DOM和CSSOM的构建是互不影响,井水不犯河水,但是一旦引入了JavaScript,CSSOM也开始阻塞DOM的构建,只有CSSOM构建完毕后,DOM再恢复DOM构建。

❓这是什么情况?

这是因为JavaScript不只是可以改DOM,它还可以更改样式,也就是它可以更改CSSOM。因为不完整的CSSOM是无法使用的,如果JavaScript想访问CSSOM并更改它,那么在执行JavaScript时,必须要能拿到完整的CSSOM。所以就导致了一个现象,如果浏览器尚未完成CSSOM的下载和构建,而我们却想在此时运行脚本,那么浏览器将延迟脚本执行和DOM构建,直至其完成CSSOM的下载和构建。也就是说,在这种情况下,浏览器会先下载和构建CSSOM,然后再执行JavaScript,最后在继续构建DOM。

❓遇到 import abc from ‘./abc.js’ 文件,会立即执行嘛?

❓疑问?1、DOM树构建完成,javascript执行中可以获取 DOM 嘛,还没有渲染阶段?

解答:1)、js文件在head中引入,获取不到当前的DOM,更不能操作DOM,因为DOM树还没有构建完成。
2)、js文件在body中引入,可以操作DOM,因为js执行之前DOM树中的

标签构建完成,CSSDOM构建完成,

❓疑问?2、获取远程的js文件,延迟会阻塞html的渲染,请求回来会解析js文件,但不会执行js文件,需要等CSSDOM解析构建完成,就可以执行js文件,等js文件执行完,就开始构建DOM树,最后布局跟绘制

1)、js文件执行会阻塞HTML的DOM 树构建,也会阻塞 DOM + CSSDOM 渲染(js文件在head阻塞,js文件在body文件后不会阻塞)
2)、js 跟 css 文件,看服务器谁返回的快,就先解析哪个,但是js文件执行,需要等CSSDOM解析、构建完成((因为style文件是pink,style1是green,不可能让屏幕一闪一闪的))
3)、js文件在head标签中引入,是无法操作DOM的,因为DOM还没渲染
4)、
( js文件跟css文件解析看服务器返回的资源,哪个时间快先解析哪个,但是js文件不会执行)js文件会阻止 DOM 构建,但是js的执行需要等 CSSDOM 构建完成,执行js文件(如果文件放在html页面的顶端)是无法获取DOM的,

❓疑问?3、js文件执行过程中,可以操作DOM嘛?

可以,具体看 js文件的位置,如果在 head标签中,不可以,DOM还没有构建,如果js文件在body中,p标签之后,那么可以操作DOM

<!DOCTYPE html>
<html lang="en">
<head>
  <!-- <link href="/style1.css" rel="stylesheet" type="text/css"> -->
  <!-- <script src="/settingPage.js"></script> -->
  <link href="/style.css" rel="stylesheet" type="text/css">
  <title>浏览器请求的HTML资源</title>
  <!-- 疑问1 疑问3?可以操作DOM吗?
    1、在此处引入js文件,开始解析、执行,会阻塞HTML中DOM树构建
    2、在head引入js文件不能操作DOM,因为构建DOM树被阻塞了
    3、js执行,阻塞 DOM、CSSDOM 树的构建,具体看实际情况,如果在 p 标签之后在执行js文件,那么页面已经开始渲染了,在head中执行js文件,肯定是要等 CSSDOM 构建完成,才会执行(因为style文件是pink,style1是green,不可能让屏幕一闪一闪的)
    4、不可以操作DOM,因为 p 标签还没有构建
  -->
  <!-- <script src="/settingPage.js"></script>
  <link href="/style2.css" rel="stylesheet" type="text/css"> -->
</head>
<body>
  <p id="ppp">Hello <span>web performance</span>students!<p>
  <!-- <div><img src="awesome-photo.jpg"></div> -->

  <!-- 疑问2 ?
    1、js文件执行过程中,DOM树 p 标签构建完成,CSSDOM也构建完成,不影响页面 DOM树的构建跟渲染(因为js会阻塞DOM树构建,DOM树可以解析一部分,构建一部分),
   -->
  <script src="/settingPage.js"></script>
  <!-- 疑问4 ?
     CSS 跟 js代码,js在后,js执行最终的样式, 
      js在 P标签之前,获取不到DOM。
      比如: <p id="ppp" style="display:none;"></p>
      <script>
          const elm = document.getElementById('ppp')
          elm.style.display = 'block'
          elm.style.background = '#000'
      </script>
   -->
</body>
</html>

❓疑问?4、渲染树中设置标签display:none,js代码中删除style,会怎么执行?

CSS 跟 js代码,js在后,js执行最终的样式,
js在 P标签之前,获取不到DOM。

  比如: <p id="ppp" style="display:none;"></p>
  <script>
      const elm = document.getElementById('ppp')
      elm.style.display = 'block'
      elm.style.background = '#000'
  </script>

页面加载过程(请看第一篇文章,浏览器输入URL会发生什么)

在这里插入图片描述

在介绍浏览器渲染过程之前,我们简明扼要介绍下页面的加载过程,有助于更好理解后续渲染过程。

要点如下:

  • 浏览器根据 DNS 服务器得到域名的 IP 地址
  • 向这个 IP 的机器发送 HTTP 请求
  • 服务器收到、处理并返回 HTTP 请求
  • 浏览器得到返回内容

例如在浏览器输入https://baidu.com,然后经过 DNS 解析,baidu.com对应的 IP 是36.36.36.36(不同时间、地点对应的 IP 可能会不同)。然后浏览器向该 IP 发送 HTTP 请求。
服务端接收到 HTTP 请求,然后经过计算(向不同的用户推送不同的内容),返回 HTTP 请求,返回的文件内容如下,以HTML举例:
(整个页面:HTML/XHTML/SVG、样式:CSS、功能性代码:JavaScript)
在这里插入图片描述
其实就是一堆 HMTL 格式的字符串,因为只有 HTML 格式浏览器才能正确解析,这是 W3C 标准的要求。接下来就是浏览器的渲染过程。

在这里插入图片描述

浏览器渲染过程 (DOM、CSSOM、CSSComputed、Layout–布局、Layer–分层、Paint–绘制、Tiling–分块、Raster–光栅、Draw–画)

  • D、C、cssComputed、布、分、绘 、分、栅、画

比如,对于这段代码,如何解析,生成DOM树🌲🌲🌲

<!DOCTYPE html>
<html>
  <head>
    <meta name="viewport" content="width=device-width,initial-scale=1">
    <link href="style.css" rel="stylesheet">
    <title>Critical Path</title>
  </head>
  <body>
    <p>Hello <span>web performance</span> students!</p>
    <div><img src="awesome-photo.jpg"></div>
  </body>
</html>

1、解析 HTML - Parse HTML Document Object Model(就好建房子,画框架图)

在这里插入图片描述

在这里插入图片描述

1)、Bytes—>Characters过程:浏览器从磁盘或网络读取HTML的原始字节,并根据文件的指定编码(例如 UTF-8)将它们转换成字符串。**;

2)、Tokens过程:将字符串转换成Token,例如:、等。Token中会标识出当前Token是“开始标签”或是“结束标签”亦或是“文本”等信息。**

疑问? 这时候你一定会有疑问,节点与节点之间的关系如何维护?

事实上,这就是Token要标识“起始标签”和“结束标签”等标识的作用。例如“title”Token的起始标签和结束标签之间的节点肯定是属于“head”的子节点。
在这里插入图片描述
上图给出了节点之间的关系,例如:“Hello”Token位于“title”开始标签与“title”结束标签之间,表明“Hello”Token是“title”Token的子节点。同理“title”Token是“head”Token的子节点。

3)、Nodes根据token序列分析语法,得到一个个节点node;**

1.1、最后根据node序列,分析并构建DOM树(建房子,框架模型已生成)。

事实上,构建DOM的过程中,不是等所有Token都转换完成后再去生成节点对象,而是一边生成Token一边消耗Token来生成节点对象。换句话说,每个Token被生成后,会立刻消耗这个Token创建出节点对象。注意:带有结束标签标识的Token不会创建节点对象。

在这里插入图片描述

2、 构建CSSOM树(解析 HTML - Parse HTML CSS Object Model)---- (建房子,装饰材料,油漆,板砖,门)

当浏览器接收到一段CSS,浏览器首先要做的是识别出Token,然后构建节点并生成CSSOM。
在这里插入图片描述

在这一过程中,浏览器会确定下每一个节点的样式到底是什么,并且这一过程其实是很消耗资源的。因为样式你可以自行设置给某个节点,也可以通过继承获得。在这一过程中,浏览器得递归 CSSOM 树,然后确定具体的元素到底是什么样式。
注意:CSS匹配HTML元素是一个相当复杂和有性能问题的事情。所以,DOM树要小,CSS尽量用id和class,千万不要过渡层叠下去。

假如somestyle.css文件的内容如下:

body { font-size: 16px }
p { font-weight: bold }
span { color: red }
p span { display: none }
img { float: right }

1)、根据node序列,生成DOM树。

在这里插入图片描述

3、 样式计算 - Recalculate Style(当我们生成 DOM 树和 CSSOM 树以后,就需要将这两棵树组合为渲染树。)-----(渲染后还不能渲染,只是图纸有了,需要买多少砖,搬砖,门,窗户)

在这里插入图片描述

在这一过程中,不是简单的将两者合并就行了。渲染树只会包括需要显示的节点和这些节点的样式信息,如果某个节点是 display: none 的,那么就不会在渲染树中显示。

4、 布局 - Layout (布局完成后,开始建房子,搭建主体,留门框,窗口)

在这里插入图片描述
在这里插入图片描述
在这里插入图片描述

5、分层 - Layer

在这里插入图片描述
点击F12,检查,找到图层(Layer),然后就可以看到相关的内容,如果要单独设置图层,使用CSS属性:will-change: transform;慎重使用。
在这里插入图片描述

6、绘制 - Paint

主线程会为每个层单独产生绘制指令集,用于描述这一层的内容该如何画出来。在这里插入图片描述
在这里插入图片描述

7、分块 - Tiling

  • 完成 绘制后,主线程将每个图层的绘制信息提交给合成线程,剩余工作将由合成线程完成。
  • 合成线程首先对每个图层进行分块,将其划分为更多的小区域。
  • 它会从线程池中拿取多个线程来完成分块工作。
    在这里插入图片描述
    ![在这里插入图片描述](https://img-blog.csdnimg.cn/direct/205da11781c246b0bb575a686a629aed.png

8、光栅化 - Raster

  • 光栅化是将页面上的 图形、文本和其他可见元素 转换为像素的过程。
  • 合成线程会将块信息交给 GPU 进程,以极高的速度完成光栅化。
  • GPU 进程会开启多个线程来完成光栅化,并且优先处理靠近视口区域的块。
  • 光栅化的结果,就是一块一块的位图
    在这里插入图片描述
    在这里插入图片描述

9、 画 - Draw(把渲染树以像素的形式绘制在页面)

在这里插入图片描述

在这里插入图片描述
在这里插入图片描述

在这里插入图片描述

文章引出以下几点

1.async和defer的作用是什么?有什么区别?

2.为什么操作 DOM 慢

  • 把 DOM 和 JavaScript 各自想象成一个岛屿,它们之间用收费桥梁连接。——《高性能 JavaScript》 JS 是很快的,在

  • JS 中修改 DOM 对象也是很快的。在JS的世界里,一切是简单的、迅速的。但 DOM 操作并非 JS

  • 一个人的独舞,而是两个模块之间的协作。 因为 DOM 是属于渲染引擎中的东西,而 JS 又是 JS 引擎中的东西。当我们用 JS 去操作

  • DOM 时,本质上是 JS 引擎和渲染引擎之间进行了“跨界交流”。这个“跨界交流”的实现并不简单,它依赖了桥接接口作为“桥梁”(如下图)。
    在这里插入图片描述

  • 过“桥”要收费——这个开销本身就是不可忽略的。我们每操作一次

  • DOM(不管是为了修改还是仅仅为了访问其值),都要过一次“桥”。过“桥”的次数一多,就会产生比较明显的性能问题。因此“减少 DOM 操作”的建议,并非空穴来风。

3.你真的了解回流和重绘吗

在这里插入图片描述

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

  • 当进行了会影响布局树的操作后,需要重新计算布局树,会引发 layout。

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

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

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

  • 在这里插入图片描述

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

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

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

4、性能优化策略

❓为什么 transform 的效率高?

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

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

  • 在这里插入图片描述

引用以下文章:

1、作者:浪里行舟文章地址
2、作者:读行笔记文章地址
3、作者:技术蛋老师 B站视频地址
4、渡一教育

  • 2
    点赞
  • 1
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
回答: 面试题中提到了几个问题,我将逐个回答。 1. eval的功能是什么? eval的功能是将对应的字符串解析成JS代码并运行。然而,应该避免使用eval,因为它不安全且非常消耗性能。[1] 2. SVG和TWO.JS是什么? SVG是一种用于描述二维矢量图形的XML标记语言,可以被引擎抓取并支持事件绑定。TWO.JS是一个SVG的JavaScript库,用于绘制二维图形。它支持三种格式:SVG、Canvas和WebGL。[2] 3. 什么是伪数组? 伪数组是一种类数组对象,它具有类似数组的特性,如具有length属性和按索引方式存储数据,但不具有数组的方法(如push、pop等)。可以通过将伪数组转化为标准数组来使用数组的方法,例如使用Array.prototype.slice.call(伪数组名称)。[3] 4. JS中实现sleep的三种方式是什么? 第一种方式是使用Promise和setTimeout实现,第二种方式是使用callback回调,第三种方式是使用循环来实现。具体代码可以参考上述引用[3]中的示例。[3] 5. 如何准确判断一个变量是数组类型? 可以使用instanceof运算符来判断一个变量是否为数组类型,例如使用arr instanceof Array。[33] 6. 从输入URL到得到HTML的过程是什么? 这个过程涉及到网络请求和浏览器渲染过程。大致步骤如下:首先,浏览器会解析URL并发送HTTP请求到服务器;服务器接收请求并返回HTML文件;浏览器接收到HTML文件后,会解析HTML并构建DOM树;同时,浏览器还会解析CSS文件并构建CSSOM树;接下来,浏览器DOM树和CSSOM树合并成渲染树;最后,浏览器根据渲染树来绘制页面并显示在屏幕上。[34] 希望以上回答能够解决你的问题。如果还有其他疑问,请随时提问。
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值