浏览器执行渲染原理

news2025/1/24 22:28:22

一、事件循环

事件循环(Event Loop)是JavaScript的执行环境的核心概念之一,它负责处理JavaScript中的异步操作和执行顺序。事件循环使得JavaScript能够在单线程上有效地处理并发,同时保持编程模型的简单性。

以下是事件循环的一些关键点:

  1. 调用栈(Call Stack): 调用栈是JavaScript执行环境的一部分,用于跟踪当前正在执行的函数。当一个函数被调用时,它会被添加到调用栈的顶部;当函数执行完毕后,它会从调用栈中移除。

  2. 事件队列(Event Queue): 事件队列是一个先进先出(FIFO)的数据结构,用于存储异步事件的回调函数。当异步操作完成时,它们的回调函数会被放入事件队列中等待执行。

  3. 事件循环机制: 事件循环不断地检查调用栈。当调用栈为空时,事件循环会从事件队列中取出一个回调函数,将其推入调用栈执行。这个过程会不断重复,形成循环。

  4. 宏任务(Macro Tasks)和微任务(Micro Tasks): 在JavaScript中,任务分为两种:宏任务和微任务。宏任务包括setTimeout、setInterval、I/O操作、UI交互等;微任务包括Promise的then/catch/finally等。事件循环会优先处理微任务队列中的所有任务,直到队列清空,然后再处理宏任务。

  5. 非阻塞I/O: JavaScript的异步I/O操作(如网络请求、文件读写等)不会阻塞调用栈的执行。当I/O操作完成时,相关的回调函数会被添加到事件队列中。

  6. 零延迟: 使用setImmediate(Node.js)或setTimeout(fn, 0)(浏览器)可以创建一个宏任务,在当前执行栈清空后的下一个事件循环迭代中执行。

  7. 异步编程模式: 事件循环是JavaScript异步编程的基础,它支持回调函数、Promise、async/await等异步编程模式。

  8. 浏览器和Node.js的事件循环: 浏览器和Node.js中的事件循环略有不同。浏览器事件循环遵循HTML5规范,而Node.js有自己的事件循环实现,称为libuv。

事件循环的引入,使得JavaScript能够以非阻塞的方式处理I/O操作和其他异步事件,从而提高程序的性能和响应性。理解事件循环对于编写高效的JavaScript代码和处理异步逻辑非常重要。

二、js计时器可以实现精确计时吗,为什么

JavaScript 计时器通常指的是 setTimeout()setInterval() 函数,这些函数用于在指定的毫秒数之后执行代码或重复执行代码。然而,JavaScript 计时器并不能保证实现精确计时,原因如下:

  1. 浏览器和操作系统的调度: JavaScript 是在浏览器中运行的,而浏览器的 JavaScript 引擎是单线程的,它需要处理页面渲染、用户输入、网络请求等。因此,JavaScript 计时器的执行可能会被其他任务的执行所延迟。

  2. 最小时间间隔限制setTimeout 有一个最小时间间隔,通常在 4 到 16 毫秒之间,即使指定的时间小于这个间隔,实际的延迟也会被增加到这个最小值。

  3. 执行栈: JavaScript 运行在一个执行栈上,如果当前执行栈非常繁忙,计时器的回调函数将被推迟执行,直到执行栈被清空。

  4. 长任务阻塞: 如果一个计时器的回调函数执行了一个非常耗时的任务,那么下一个回调函数的执行时间将会被推迟。

  5. 页面不活跃: 当浏览器标签页不活跃或被最小化时,为了节能和减少CPU使用,浏览器可能会延迟计时器的回调执行。

  6. 高精度时间测量: 尽管 setTimeoutsetInterval 不能保证精确计时,但 JavaScript 提供了 performance.now() 方法,它可以提供高精度的时间测量,通常用于跟踪和性能分析。

  7. Web Workers: 对于需要非常精确计时的应用,可以考虑使用 Web Workers,它允许长时间运行的脚本在后台线程上运行,从而不干扰用户界面的响应性。

  8. requestAnimationFrame(): 对于需要与浏览器的刷新率同步的动画和视觉效果,可以使用 requestAnimationFrame(),它提供了更好的性能和更接近于显示器刷新率的同步。

  9. 服务端处理: 如果精确计时是绝对必要的,并且可以容忍网络延迟,可以考虑在服务器端进行计时,然后通过网络请求与客户端同步。

因此,如果需要进行精确的时间控制,JavaScript 计时器可能不是最佳选择。在大多数情况下,它们适用于简单的时间延迟和周期性任务,但对于需要高精度的场景,应该考虑上述提到的替代方案。

三、浏览器渲染原理

面试问题:

浏览器渲染页面是一个复杂的过程,涉及多个步骤。以下是浏览器渲染页面的一般步骤:

  1. 解析 HTML: 浏览器开始解析HTML文档,构建DOM(文档对象模型)树。DOM树是由HTML元素组成的树状结构,每个节点代表一个HTML元素。

  2. 解析 CSS: 接着,浏览器解析CSS样式表,构建CSSOM(CSS对象模型)树。CSSOM树包含了所有CSS规则,用于确定页面的视觉样式。

  3. 构建渲染树: 浏览器将DOM树和CSSOM树结合起来,创建渲染树。渲染树是一个包含所有需要显示的元素的树状结构,它考虑了CSS样式和HTML结构,但不包括那些在CSS中通过display: none或其他方式隐藏的元素。

  4. 布局(Layout)/ 重排(Reflow): 浏览器使用渲染树来计算每个节点的几何信息,即每个元素在屏幕上的确切位置和尺寸。这个过程称为布局或重排

  5. 绘制(Painting): 最后,浏览器遍历布局好的渲染树,并实际在屏幕上绘制各个节点。这个过程称为绘制或重绘

  6. 合成(Compositing): 现代浏览器使用GPU加速渲染,会将多个层合成到屏幕上。合成是GPU将页面的各个部分组合在一起的过程,以生成最终的像素输出。

  7. 优化: 为了提高性能,浏览器会尝试最小化布局、绘制和合成的次数。例如,浏览器可能会使用层合成来减少重绘和重排。

  8. JavaScript 执行: JavaScript代码可能会改变DOM或CSSOM,这会导致浏览器重新执行上述渲染步骤。例如,JavaScript可以动态添加或删除元素,更改样式或响应用户事件。

  9. 事件处理: 浏览器还需要处理用户交互,如点击、滚动和输入,这些事件可能触发JavaScript代码的执行,进一步影响DOM、CSSOM和渲染树。

为了提高性能,浏览器使用了一些优化技术,如:

  • 防抖(Debouncing)和节流(Throttling):减少JavaScript事件处理函数的调用频率。
  • 虚拟滚动:只渲染视口附近的内容,而不是渲染整个长页面的所有内容。
  • 懒加载:延迟加载那些不在视口中的图片或脚本。

理解浏览器的渲染原理对于优化网站性能至关重要,因为它可以帮助开发者识别性能瓶颈,并采取措施减少页面加载时间和提高响应速度。

注意js特殊

生成指令

四、包含块

在CSS布局中,包含块(Containing Block)是指用于确定元素尺寸和位置的参考矩形。每个元素都有自己的包含块,它可能是由HTML元素的尺寸和位置确定的,也可能是由CSS属性如positionwidthheighttoprightbottomleft确定的。

以下是一些关于包含块的重要概念:

  1. 初始包含块: 每个HTML页面都有一个初始包含块,通常对应于浏览器窗口的可视区域。对于HTML文档的根元素<html>,其包含块就是整个浏览器的视口(viewport)。

  2. 定位方案的包含块: 对于非定位元素(即没有设置position:absoluteposition:fixedposition:sticky的元素),其包含块由其最近的块级祖先元素的content区域决定。对于定位元素(position:absoluteposition:fixed),其包含块是其最近的非静态定位的祖先元素。

  3. 绝对定位的包含块: 对于position: absolute的元素,包含块是其最近的非静态定位的祖先元素。如果没有这样的祖先元素,则包含块是初始包含块,即浏览器的视口。

  4. 固定定位的包含块: 对于position: fixed的元素,包含块始终是浏览器的视口。

  5. 替换元素的包含块: 替换元素(如图片、视频等)的包含块由其固有尺寸决定,即使它们有外边距(margin)。

  6. 宽度和高度: 元素的widthheight属性可以指定其包含块的尺寸,除非它们是替换元素,替换元素的尺寸由其固有尺寸决定。

  7. 视口单位: 使用视口单位(如vwvhvminvmax)时,包含块是浏览器的视口。

  8. CSS Grid 和 Flexbox: 在CSS Grid和Flexbox布局中,子元素的包含块是由其父级Grid容器或Flex容器创建的。

包含块的概念对于理解元素如何在页面上定位以及它们如何相互影响至关重要。正确地理解包含块可以帮你更好地控制元素的布局和尺寸。

五、优化有关问题:

Reflow(重排)

在浏览器的渲染过程中,Reflow(重排)或Layout(布局)是一个重要的概念。它指的是浏览器重新计算并调整页面元素的位置和尺寸的过程。当页面的DOM结构变化或样式发生变化时,浏览器需要重新进行布局以反映这些更改。

以下是一些可能触发重排的常见操作:

  1. 修改DOM元素:添加、删除或修改DOM树中的元素。
  2. 改变元素样式:更改元素的样式属性,如widthheightmarginpaddingborder等。
  3. 窗口大小变化:用户调整浏览器窗口大小或在不同设备之间切换。
  4. 元素尺寸变化:图片加载完成后,其尺寸可能会改变,导致周围元素重新布局。
  5. 计算样式属性:使用getComputedStyle()window.getComputedStyle()等方法可以触发重排。
  6. 查询元素的尺寸和位置:读取元素的offsetTopoffsetLeftoffsetWidthoffsetHeightscrollTopscrollLeftclientTopclientLeftclientWidthclientHeight等属性。

重排是一个相对昂贵的操作,因为它涉及以下步骤:

  1. 计算样式:浏览器需要重新应用CSS样式到DOM元素上。
  2. 构建渲染树:浏览器需要重新构建渲染树,这可能涉及到合并DOM和CSSOM。
  3. 布局:浏览器需要重新计算页面元素的位置和尺寸。
  4. 绘制:浏览器需要重新绘制受影响的元素到屏幕上。

为了提高性能,浏览器会尝试最小化重排的发生次数。一些优化重排的技巧包括:

  • 批量修改:使用DocumentFragmentdisplay: none隐藏的元素进行批量DOM操作,然后再将它们添加到文档中。
  • 避免复杂的选择器:使用简单的CSS选择器,减少浏览器计算样式所需的时间。
  • 避免使用硬编码的尺寸:使用相对单位(如%emvwvh)而不是绝对单位(如px)。
  • 使用transformopacity动画:这些属性的更改可以触发浏览器的合成层(Compositing layers),从而减少重排和重绘。

理解重排的概念对于优化网页性能至关重要,因为频繁的重排会导致性能下降,特别是在复杂或动画丰富的页面上。

repaint重绘

在浏览器的渲染过程中,Repaint(重绘)是指当页面中的某些元素的外观或风格发生变化,但这些变化不会影响到元素的布局或几何属性时,浏览器需要重新绘制这些元素的过程。

重绘与重排(Reflow)紧密相关,但它们关注页面渲染的不同方面:

  • 重排:浏览器需要重新计算页面元素的位置和尺寸,这通常发生在DOM结构变化或元素的几何属性发生变化时。
  • 重绘:浏览器需要重新绘制页面元素的外观,这通常发生在元素的视觉属性发生变化时,而这些变化并不影响元素的布局。

以下是一些可能触发重绘的操作:

  1. 颜色变化:更改元素的背景色、前景色、边框颜色等。
  2. 背景变化:更改元素的背景图片、背景渐变等。
  3. 文本样式变化:更改文本的字体、字重、字体大小、行高、文本颜色等。
  4. 阴影和边框样式变化:更改元素的阴影、边框样式、边框宽度等。
  5. 透明度变化:更改元素的透明度,如使用CSS的opacity属性。
  6. CSS动画:某些CSS动画,如opacitytransform等,可能会触发重绘但不一定触发重排。

与重排相比,重绘通常对性能的影响较小,因为重绘不需要重新计算元素的布局。然而,如果重绘涉及的元素非常复杂或数量很大,它仍然可能导致性能问题。

为了优化性能,可以采取以下措施:

  • 减少重绘次数:避免在循环或重绘区域内频繁修改元素的视觉属性。
  • 使用CSS动画:CSS动画可以利用浏览器的优化,如使用transformopacity属性可以触发GPU加速渲染。
  • 利用合成层:某些浏览器提供了合成层(Compositing layers),可以将某些元素的渲染独立于主线程,从而提高性能。

理解重绘的概念对于优化网页性能非常重要,尤其是在创建动画和视觉效果时。通过减少不必要的重绘,可以提高页面的渲染性能和响应速度。

只改变了style

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

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

相关文章

智慧油田三维电子沙盘系统

深圳易图讯科技(www.3dgis.top)智慧油田三维电子沙盘系统采用三维GIS、大数据、云计算、虚拟现实、物联网、AI等前沿技术&#xff0c;支持无人机航拍、高清卫星影像、DEM高程数据、矢量数据、无人机倾斜摄像、BIM模型、点云、城市白模、等高线、标高点等数据融合和切换&#xf…

SCI论文发表:寻找论文选题的7种方法 (建议收藏)!

我是娜姐 迪娜学姐 &#xff0c;一个SCI医学期刊编辑&#xff0c;探索用AI工具提效论文写作和发表。 近期有好几个同学问娜姐关于论文选题的问题&#xff1a;什么样的选题是好的选题-有价值好发表。这篇来分享7种寻找有效选题的方法。 在我们寻找论文选题的过程中&#xff0c;…

猫头虎分享已解决Bug || **Babel转换器下载问题** Failed to resolve babel-loader dependency`

猫头虎分享已解决Bug &#x1f42f; || Babel转换器下载问题 &#x1f6ab;Failed to resolve babel-loader dependency 博主猫头虎的技术世界 &#x1f31f; 欢迎来到猫头虎的博客 — 探索技术的无限可能&#xff01; 专栏链接&#xff1a; &#x1f517; 精选专栏&#xff1a…

mapreduce | 自定义Partition分区(案例1)

1.需求 将学生成绩&#xff0c;按照各个成绩降序排序&#xff0c;各个科目成绩单独输出。 # 自定义partition 将下面数据分区处理&#xff1a; 人名 科目 成绩 张三 语文 10 李四 数学 30 王五 语文 20 赵6 英语 40 张三 数据 50 李四 语文 10 张三 英语 70 李四 英语…

BGP大型实验

一.实验拓扑 二.实验要求 1.As1中存在两个环回个地址为192.168.1.0/24&#xff0c;该地址不能在任何协议中宣告&#xff1b; AS3中存在两个环回&#xff0c;一个地址为192.168.2.0/24&#xff0c;不能在任何协议中宣告&#xff0c;最终要求这两个环回可以ping通&#xff1b; …

【备战软考(嵌入式系统设计师)】12 - 嵌入式系统总线接口

我们嵌入式系统的总线接口可以分为两类&#xff0c;一类是并行接口&#xff0c;另一类是串行接口。 并行通信就是用多个数据线&#xff0c;每条数据线表示一个位来进行传输数据&#xff0c;串行接口就是一根数据线可以来一位一位地传递数据。 从上图也可以看出&#xff0c;并行…

理解JavaScript递归

什么是递归 程序调用自身的编程技巧称为递归&#xff08;recursion&#xff09; 递归的基本思想是将一个复杂的问题分解成更小、更易于管理的子问题&#xff0c;这些子问题与原始问题相似&#xff0c;但规模更小。 递归的要素 基本情况&#xff08;Base Case&#xff09;&…

【linux】linux工具使用

这一章完全可以和前两篇文件归类在一起&#xff0c;可以选择放一起看哦 http://t.csdnimg.cn/aNaAg http://t.csdnimg.cn/gkJx7 拖更好久了&#xff0c;抱歉&#xff0c;让我偷了会懒 1. 自动化构建工具 make , makefile make 是一个命令&#xff0c;makefile 是一个文件&…

【RAG 论文】FiD:一种将 retrieved docs 合并输入给 LM 的方法

论文&#xff1a; Leveraging Passage Retrieval with Generative Models for Open Domain Question Answering ⭐⭐⭐⭐ EACL 2021, Facebook AI Research 论文速读 在 RAG 中&#xff0c;如何将检索出的 passages 做聚合并输入到生成模型是一个问题&#xff0c;本文提出了一…

VMware虚拟机故障:“显示指定的文件不是虚拟磁盘“,处理办法

一、故障现象 由于虚拟机宕机&#xff0c;强制重新启动虚拟机后显示错误&#xff0c;没有办法启动虚拟机。 虚拟机有快照&#xff0c;执行快照还原&#xff0c;结果也不行&#xff0c;反复操作&#xff0c;在虚拟机文件目录出现很多莫名文件 二、故障原因 根据故障提示&#…

浅谈@Controller注解和其他四大注解的区别

各位大佬光临寒舍&#xff0c;希望各位能赏脸给个三连&#xff0c;谢谢各位大佬了&#xff01;&#xff01;&#xff01; 目录 1.Spring五大注解的使用约定 2.Controller注解的特别之处 3.总结 1.Spring五大注解的使用约定 Spring的五大注解&#xff08;Controller&#x…

flutter自定义日期选择器按日、按月、自定义开始、结束时间

导入包flutter_datetime_picker: 1.5.0 封装 import package:atui/jade/utils/JadeColors.dart; import package:flutter/cupertino.dart; import package:flutter/material.dart; import package:flutter_datetime_picker/flutter_datetime_picker.dart; import package:flut…

040——移植数据库sqlite3到i.mx6ull

目录 一、下载 二、移植数据库 三、测试sqlite3 一、下载 SQLite Download Page 暂时先下载最新版的试试&#xff0c;我们以前其实在ubuntu上直接使用过 嵌入式数据库sqlite3_常见的嵌入式数据库-CSDN博客 当时我把常用的操作和怎么使用记录下来了 现在把他移植到开发板…

SamFirm Reborn 0.3.6.8三星固件下载器 汉化版

介绍 在三星手机的维护和升级过程中&#xff0c;固件的获取往往成为了一个难题。幸运的是&#xff0c;有一群热爱技术的开发者们&#xff0c;他们开发了各种工具以简化这个过程。今天&#xff0c;我们要介绍的是一款名为SamFirm Reborn 0.3.6.8的三星固件下载器的汉化版。它旨在…

Java8 ConcurrentHashMap 存储、扩容源码阅读

文章目录 1. 概述2. 入门实例3. 属性4. 核心方法4.1 put4.2 initTable4.3 transfer4.4 sizeCtl4.5 sizeCtl bug 1. 概述 ConcurrentHashMap 是线程安全且高效的 HashMap。 HashMap 可以看下我这篇 传送门 。 2. 入门实例 public class MyStudy {public static void main(St…

什么是数据平台——企业构建Data+AI的基础数据底座需要的决策参考

什么是数据平台 标准的解释是这样的 Wikipedia A data platform usually refers to a software platform used for collecting and managing data, and acting as a data delivery point for application and reporting software. 数据平台是指将各类数据进行整合、存储、处…

实现字符串比较函数(C语言)

一、N-S流程图&#xff1b; 二、运行结果&#xff1b; 三、源代码&#xff1b; # define _CRT_SECURE_NO_WARNINGS # include <stdio.h>int main() {//初始化变量值&#xff1b;int i, result;char s1[100], s2[100];//填充数组&#xff1b;printf("请输入数组s1的…

Docker 直接运行一个 Alpine 镜像

由于镜像很小&#xff0c;下载时间往往很短&#xff0c;读者可以直接使用 docker run 指令直接运行一个 Alpine 容器&#xff0c;并指定运行的 Linux 指令&#xff0c;例如&#xff1a; PS C:\Users\yhu> docker run alpine echo 123 Unable to find image alpine:latest lo…

docker八大架构之应用数据分离架构

数据分离架构 什么是数据分离架构&#xff1f; 数据分离架构是指应用服务&#xff08;应用层&#xff09;和数据库服务&#xff08;数据层&#xff09;使用不同的服务器来进行操作&#xff0c;如下边的两个图所示。当访问到应用层后&#xff0c;如果需要获取数据会进行访问另…

【Qt 学习笔记】Qt常用控件 | 布局管理器 | 水平布局Horizontal Layout

博客主页&#xff1a;Duck Bro 博客主页系列专栏&#xff1a;Qt 专栏关注博主&#xff0c;后期持续更新系列文章如果有错误感谢请大家批评指出&#xff0c;及时修改感谢大家点赞&#x1f44d;收藏⭐评论✍ Qt常用控件 | 布局管理器 | 水平布局Horizontal Layout 文章编号&…