DOM 更新渲染时机

问题

事件循环

  1. 进入到 script 标签,就进入到了第一次事件循环.
  2. 遇到同步代码,立即执行
  3. 遇到宏任务,放入到宏任务队列里.
  4. 遇到微任务,放入到微任务队列里.
  5. 执行完所有同步代码
  6. 执行微任务代码
  7. 微任务代码执行完毕,本次队列清空
  8. 更新 DOM 渲染
  9. 寻找下一个宏任务,重复步骤1

那么在宏任务或微任务中操作 DOM 之后,再获取 DOM 的最新状态或位置,这时 DOM 还没渲染为什么能获取到。

更新 DOM 渲染

首先需要用改变 DOM 才会触发更新 DOM 渲染。如下:

这段代码是同步执行的,因此

能打印最新的值。

此时页面并没有渲染出最新的内容。

这段代码运行效果如下:

这里用到了 alert ,在 chrome 中 alert 会暂停任务执行,相应的事件循环不得运行进一步的任务,并且当前正在运行的任务中的任何脚本都必须阻塞。 因此在 alert 信息弹出,页面显示"测试",说明页面没有渲染。而提示框提示的信息是"测试"。

我们修改下添加宏任务和微任务:

运行效果如下:

微任务提示框时页面没有重新渲染。

宏任务提示框时页面渲染了。

页面渲染不是同步渲染的,而是在一个事件循环微任务执行完之后才会渲染,也就是在一个事件循环内缓存宏任务和微任务的 DOM 操作,最后才一起渲染到页面上。

我们用 document 的 API 修改 DOM 时,在用 document 的 API 获取 DOM 的属性自然也就是最新值。

那该元素在页面的位置和尺寸大小能不能获取到最新的呢?

页面渲染主要分为两个阶段:Reflow(回流)、Repaint(重绘)。

Reflow(回流),有的叫它重排,这个阶段主要是重新触发了 Layout(布局),重新计算元素的所有框位置。

Repaint(重绘),将元素样式绘制在屏幕上。

如果页面的是在微任务之后执行重新渲染,那么在宏任务中修改尺寸时没有触发 Layout(布局)应该获取不到最新尺寸的,我们写个例子看看是不是这样:

运行结果:

运行结果可以看到可以获取到最新的尺寸,我们用 devtool 工具监听页面的渲染过程。

图中看到执行了两个 alert,我们把两个 alert 之间的部分放大看看:

两个 alert 直接触发了 Layout(布局)。

也就是当获取 DOM 尺寸的时候强制执行了 Layout(布局) 后再获取 DOM 的尺寸,但并没有触发重绘,只是触发了布局。

总结

  • DOM 的修改时同步修改。
  • Reflow(回流),通常浏览器会等到宏任务和微任务操作DOM执行完之后再触发它,但是一些 DOM 方法比如获取尺寸 offsetXXX 等会强制触发同步 Relow(回流),以便获取最新值,但不会强制触发 Repaint(重绘)。
  • Repaint(重绘),将元素样式绘制在屏幕上,大多数浏览器会等待下一次屏幕刷新,不同浏览器重绘规则有区别。
相关推荐
teeeeeeemo7 分钟前
JS数据类型检测方法总结
开发语言·前端·javascript·笔记
木木黄木木7 分钟前
HTML5 火焰字体效果教程
前端·html·html5
云墨-款哥的博客8 分钟前
失业学习-前端工程化-webpack基础
前端·学习·webpack
MAOX78911 分钟前
基于python的web系统界面登录
前端·python
懒大王、12 分钟前
Vue添加图片作为水印
前端·javascript·vue.js
Junerver16 分钟前
如何在Jetpack Compose中轻松的进行表单验证
前端·kotlin
3Katrina19 分钟前
《JavaScript this 指向深度剖析:从基础到复杂场景实战》
前端·javascript
岩柏22 分钟前
在vue项目中添加stylelint
前端
暖苏24 分钟前
Vue.js第一节
前端·javascript·css·vue.js·ecmascript
前端服务区28 分钟前
NodeJS文件流
前端