浏览器执行渲染原理

一、事件循环

事件循环(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.mzph.cn/bicheng/11624.shtml

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

相关文章

使用Eigen将经纬度、高程、偏北角转成变换矩阵

目录 1、前言 2、示例 3、代码解析 4、垂直于给定点的切平面变换 5、代码解析 1、前言 在地球表面进行刚体变换时候&#xff0c;要将具有经纬度、高程和偏北角的坐标信息转换为变换矩阵表达&#xff0c;首先需要了解坐标系之间的转换关系。 通常&#xff0c;我们会将经纬…

AI2024(64bit) Adobe Illustrator 软件安装包下载

AI2024(64bit) Adobe Illustrator 软件安装包下载地址&#xff1a; 百度网盘下载https://pan.baidu.com/s/1C10-2JVN1rxFF5VFRuV2Yw?pwdSIMS 在创意设计的浩瀚宇宙中&#xff0c;Adobe Illustrator 2024如同一颗璀璨新星&#xff0c;以其无与伦比的创新功能和优化体验&#x…

快团团新人怎么找供货团长?免费教程一学就会!

作为快团团的新手&#xff0c;想要寻找供货团长&#xff0c;可以按照以下步骤进行&#xff1a; 打开微信&#xff1a;首先&#xff0c;在您的手机上打开微信应用。 搜索快团团&#xff1a;在微信顶部的搜索框中输入“团长运营之家”&#xff0c;选择出现的“团长运营之家”公号…

搭建网站式个人网盘-超仿Windows界面

搭建网站式个人网盘-超仿Windows界面 效果图部分源码领取源码下期更新预报 效果图 一款网站式个人网盘源码-Windows界面相等于一个网站式电脑可以放照片&#xff0c;视频-支持在线播放&#xff08;你懂的&#xff09;我觉得式一款很不错的个人网盘提醒&#xff1a;千万不要升级…

【STM32HAL库】DAC输出0-3.3v

一、简要介绍一下DAC DAC也有分辨率&#xff0c;转换时间&#xff0c;精度等 分辨率常见为8或12位的 转换时间F1&#xff0c;F4,F7都是3us左右&#xff0c;而H7系列是1.7us 1.DAC框图 2.数据格式&#xff08;对齐方式&#xff09; 3.触发源 4.可以发送DMA请求 注意&#xff…

【知识碎片】2024_05_13

本文记录了两道代码题【自除数】和【除自身以外数组的乘积】&#xff08;利用了前缀积和后缀积&#xff0c;值得再看&#xff09;&#xff0c;第二部分记录了关于指针数组和逗号表达式的两道选择题。 每日代码 自除数 . - 力扣&#xff08;LeetCode&#xff09; /*** Note: T…

类与对象(二)

封装 封装作为面向对象三大特性&#xff08;封装&#xff0c;继承&#xff0c;多态&#xff09;之一&#xff0c;那如何实现封装性的呢&#xff1f;就又得拿出上面的访问修饰限定符的图 public: 就是在任何地方都可以访问 protected: 涉及子类在介绍继承时详细介绍 default: …

分类预测 | Matlab实现DBO-CNN-SVM蜣螂算法优化卷积神经网络结合支持向量机多特征分类预测

分类预测 | Matlab实现DBO-CNN-SVM蜣螂算法优化卷积神经网络结合支持向量机多特征分类预测 目录 分类预测 | Matlab实现DBO-CNN-SVM蜣螂算法优化卷积神经网络结合支持向量机多特征分类预测分类效果基本描述程序设计参考资料 分类效果 基本描述 1.Matlab实现DBO-CNN-SVM蜣螂算法…

APP反抓包 - 客户端证书验证进阶(代码混淆)

1.关于混淆 在安卓开发中,对于第三方的包是可以进行混淆的,例如:OKHttp3.Http.Cert.check 被混淆后可以是a.f.c.b 形式。在安卓开发中,系统包是无法混淆的,例如:java.security.KeyStore不会被混淆。由于这种的情况的存在,再次审示我们之前的通用脚本,就会发现他是不通用…

计算机Java项目|Springboot房产销售系统

作者主页&#xff1a;编程指南针 作者简介&#xff1a;Java领域优质创作者、CSDN博客专家 、CSDN内容合伙人、掘金特邀作者、阿里云博客专家、51CTO特邀作者、多年架构师设计经验、腾讯课堂常驻讲师 主要内容&#xff1a;Java项目、Python项目、前端项目、人工智能与大数据、简…

mybatis-plus(2)

上文我们介绍完mybatis-plus的常用注解&#xff0c;现在介绍 mp的基础的yaml配置 mybatis-plus:type-aliases-package: #该位置写 数据库对应实体类的全路径global-config:db-config:id-type: auto # 全局id类型为自增长 mp同时也是支持手写sql&#xff0c;而且mapper的读取地…

如何用 OceanBase做业务开发——【DBA从入门到实践】第六期

当应用一款新的数据库时&#xff0c;除了基础的安装部署步骤&#xff0c;掌握其应用开发方法才是实现数据库价值的关键。为此&#xff0c;我们特别安排了5月15日&#xff08;周三&#xff09;的《DBA 从入门到实践》第六期课程——本次课程将带大家了解OceanBase数据库的开发流…

【爬虫】爬取股票历史K线数据写入数据库(三)

前几天有写过两篇&#xff1a; 【爬虫】爬取A股数据写入数据库&#xff08;二&#xff09; 【爬虫】爬取A股数据写入数据库&#xff08;一&#xff09; 现在继续完善&#xff0c;分析及爬取股票的历史K线数据通过ORM形式批量写入数据库。 2024/05&#xff0c;本文主要内容如下…

JavaScript数字(Number)个数学(Math)对象

目录 前言&#xff1a; Number&#xff08;数字&#xff09;对象 前言&#xff1a; nfinity(正负无穷大)&#xff1a; NaN&#xff08;非数字&#xff09;&#xff1a; Number的属性 Number的方法 构造函数 静态方法 实例方法 Math&#xff08;数学&#xff09;对象…

【数据结构】详解队列

现在我们来掌握一下队列&#xff01;如果有对往期知识有不足地方&#xff0c;可翻阅之前文章哦&#xff01; 个人主页&#xff1a;小八哥向前冲~-CSDN博客 所属专栏&#xff1a;数据结构【c语言版】_小八哥向前冲~的博客-CSDN博客 栈和队列的实现其实都是对你顺序表和链表的检验…

Dev-C++的下载和安装教程(超详细图文,小白入门)

Dev-C&#xff08;或者叫做Dev-Cpp&#xff09;是Windows环境下的一个轻量级C/C集成开发环境&#xff08;IDE&#xff09;。它是一款自由软件&#xff0c;遵守GPL许可协议分发源代码。 Dev-C集合了功能强大的源码编辑器、MingW64/TDM-GCC编译器、GDB调试器和AStyle格式整理器等…

4.uniapp+vue3项目使用vuex

文章目录 1. uniappvue3项目使用vuex1.1. main.js引入store1.2. 创建store/index.js1.3. 项目中引用1.4. 开始解决实际问题1.5. vuex和storage的区别 1. uniappvue3项目使用vuex 这篇文章&#xff0c;既是使用的教程&#xff0c;也是用来解决一个实际问题&#xff1a;uView自定…

GO语言核心30讲 实战与应用 (WaitGroup和Once,context,Pool,Map,字符编码,string包,bytes包)

原站地址&#xff1a;Go语言核心36讲_Golang_Go语言-极客时间 一、sync.WaitGroup和sync.Once 1. sync.WaitGroup 比通道更加适合实现一对多的 goroutine 协作流程。 2. WaitGroup类型有三个指针方法&#xff1a;Wait、Add和Done&#xff0c;以及内部有一个计数器。 (1) Wa…

代码随想录训练营Day 27|理论基础、力扣 77. 组合

1.理论基础 题目链接/文章讲解&#xff1a;代码随想录 视频讲解&#xff1a;带你学透回溯算法&#xff08;理论篇&#xff09;| 回溯法精讲&#xff01;_哔哩哔哩_bilibili 来自代码随想录的网站&#xff1a; void backtracking(参数) {if (终止条件) {存放结果;return;}for (…

基于springboot+vue+Mysql的音乐翻唱与分享平台

开发语言&#xff1a;Java框架&#xff1a;springbootJDK版本&#xff1a;JDK1.8服务器&#xff1a;tomcat7数据库&#xff1a;mysql 5.7&#xff08;一定要5.7版本&#xff09;数据库工具&#xff1a;Navicat11开发软件&#xff1a;eclipse/myeclipse/ideaMaven包&#xff1a;…