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(重绘),将元素样式绘制在屏幕上,大多数浏览器会等待下一次屏幕刷新,不同浏览器重绘规则有区别。
相关推荐
JohnYan6 分钟前
Bun技术评估 - 30 SSE支持
javascript·后端·bun
鹤归时起雾.12 分钟前
CSS属性继承与元素隐藏全解析
前端·css
yzx99101315 分钟前
一个嵌入式存储芯片质量评估系统的网页界面设计
开发语言·javascript·ecmascript
火星数据-Tina18 分钟前
让电竞数据实时跳动:Spring Boot 后端 + Vue 前端的完美融合实践
前端·vue.js·spring boot
fruge29 分钟前
前端可视化家庭账单:用 ECharts 实现支出统计与趋势分析
前端·javascript·echarts
IT_陈寒1 小时前
Vue3性能优化实战:5个被低估的Composition API技巧让你的应用快30%
前端·人工智能·后端
嘻嘻哈哈猿人1 小时前
从 0 到 1 实现一个支持 @ 提及用户的输入框组件(Vue3 实战)
前端·vue.js
东土也1 小时前
Vue 项目 Nginx 部署路径差异分析与部署指南
前端
云枫晖1 小时前
Vue3 响应式原理:手写实现 ref 函数
前端·vue.js
合作小小程序员小小店1 小时前
web网页开发,在线%宠物销售%系统,基于Idea,html,css,jQuery,java,ssh,mysql。
java·前端·数据库·mysql·jdk·intellij-idea·宠物