JS进阶(五)同步异步编程及浏览器的底层渲染机制

本文涉及的产品
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
简介: 浏览器渲染机制浏览器底层渲染机制一个页面从服务器访问,拿到页面源代码之后做的事情是什么?生成Dom树(DOM Tree) => 对HTML文件的处理基于HTML获取的是流文件 (进制编码)把进制编码编译为具体的字符按照令牌TOKEN进行解析 (分词/断词)生成具体的节点 (元素标签/文本节点....)按照相互的依赖关系生成一个DOM树 (节点树)

浏览器渲染机制

浏览器底层渲染机制

一个页面从服务器访问,拿到页面源代码之后做的事情是什么?

  1. 生成Dom树(DOM Tree) =>  对HTML文件的处理
  • 基于HTML获取的是流文件 (进制编码)
  • 把进制编码编译为具体的字符
  • 按照令牌TOKEN进行解析 (分词/断词)
  • 生成具体的节点 (元素标签/文本节点....)
  • 按照相互的依赖关系生成一个DOM树 (节点树)

屏幕截图 2023-07-05 233218.png

生成CSSOM Tree => 对CSS文件进行处理

屏幕截图 2023-07-05 233239.png

  1. 生成渲染树(Render Tree)
  • DOM TREE + CSSOM TREE
  • Meat、Head、Link和display:none;等这些元素是不会出现在渲染树的
  1. 布局/回流/重排 Layout
  • 按照渲染树计算出每一个元素在视口中的位置和大小
  1. 分层
  • 按照计算出来的层级进行分层
  • 定位
  • 设置透明度 rgba
  • 设置滤镜
  • 文本超过盒子大小,裁切
  • 单独计算每一层的绘制列表(具体绘制)
  1. 绘制/重绘 Painting
  • 把生成的绘制列表提交给"合成线程"
  • "合成线程"进行我们最后的绘制,呈现在浏览器的页面上
  1. 光栅化,合成

分层之前的操作都是基于GUI渲染线程来完成

进程与线程

  • 进程: 一个程序(浏览器新建一个也卡就是一个进程) 【工厂】
  • 线程: 一个进程中可能包含多个线程,每个线程可以同时做一件事情,真正的同时做一件事情,必须依赖多线程【工人】

JS中的同步异步

浏览器就是多线程 , 但是JS本身是单线程。因为浏览器本身只分配了一个线程“GUI渲染线程”运行JS代码,JS本身从本质上来讲,是不能同时做多件事的。

  • 同步:上一件事情完成,再去做下一件事情
  • 异步: 上一件事情没有完成(我们把它做一些特殊处理),下一件事情继续执行 【绝对不是同时做两件事】

浏览器生成DOM TREE/CSSOM TREE ... 的过程也是单线程(配合浏览器的多线程去完成一些事情,例如资源请求就是利用浏览器的网络线程去做的)

浏览器具体的解析过程 "GUI渲染线程"

  1. 自上而下解析完所有的HTML标签/DOM节点后,DOM TREE就生成了
  2. 解析过程中会遇到比较特殊的
  • 外链式  link href = “”
    => 浏览器会分配一个新的HTTP网络进程去加载资源文件
    => 不会阻碍DOM树的渲染
  • 内嵌式  < style > 。。。</ style >
    => 不用去请求新的资源文件了,但是此时样式没有处理,浏览器会做一个记录 ,它会等所有的CSS资源加载回来之后,按照先后顺序依次渲染CSS,从而生成CSSOM树
  • @import 'xxx.css' 导入式
    =>  虽然也是分配HTTP网络线程去加载文件,但是此时GUI渲染线程会被阻塞掉【阻碍DOM树的渲染】(只有等待资源加载回来,才会继续渲染DOM)
  1. 遇到script标签
  • 遇到内嵌js代码,会立即执行JS (阻碍dom树渲染)
  • 遇到script 外链js代码的
  • 阻碍DOM TREE的渲染,同时分配一个HTTP线程去加载资源文件,加载回来后立即执行JS ( JS 中没有采用异步,直接获取DOM元素,而DOM元素此时没有渲染,JS是获取不到的)
  • 把script放到页面底部 (先渲染DOM TREE,再执行JS,也可以获取到DOM元素了)

想要script放在前面还可以获取dom,有3种解决办法
> setTimeout(()=>{ ... }, 0) 
> window.addEventListener('DOMContentLoaded',function(){ ...   })  触发条件:DOM TREE加载完即可
> window.onload = function(){ ... }  => 触发条件是:所以dom资源都加载完成(包含DOM TREE/CSS/图片),再执行

async / defer 给script设置的属性

> async 是开辟HTTP线程加载资源文件,此时DOM TREE继续,但是资源文件一旦加载回来,停止DOM TREE,先执行JS代码(不考虑JS引用顺序,谁先加载回来谁先执行)
> defer 也是开辟HTTP线程加载资源文件,即使资源文件加载回来,也会等待DOM树渲染完成,defer效率更好,但是不兼容低版本浏览器

屏幕截图 2023-07-05 233319.png

  • 遇到img,正常情况下,老版本浏览器会阻塞dom渲染,新版浏览器虽然不会阻碍DOM渲染,但是会占用图片资源请求,会占用HTTP线程(浏览器只能同时开6-7个HTTP线程,这样的话,图片或者音视频资源加载本来就会慢一些,会影响关键(其他link/script)资源的加载), 图片资源的渲染也是比一般资源耗时间的,也会拖累渲染速度

以上性能优化的点:

1、不用@import

2、link标签放到HEAD中,尽可能提前加载资源文件,这样等DOM树渲染完,资源就加载回来了: 当代浏览器的机制越发完善,chrome的预加载扫描器通过"src","link"等属性,找到外部链接资源后预加载,避免了资源加载等待,同事实现了提前加载和执行分离

3、如果CSS代码比较少,尽可能使用内嵌式,可以减少HTTP请求,但是如果样式比较多,采用内嵌式,第一次加载HTML都会浪费很长时间,这样还不如基于link分开加载; 移动端开发都是内嵌优先(也要考虑CSS代码量)

4、尽可能减少DOM或者减少DOM的层级嵌套,以及标签语义化(当代前端开发,只把首屏结构/内容写出来,渲染知识首屏的,当首屏加载完,页面滚动的时候,在给予JS创建其他屏幕的结构和内容 => 骨架屏/SSR => 客户端股价屏,开始首屏结构都没有,只有一个loading或者展位图)

5、图片合并(sripte) / BASE64  / iconfont / svg / http2 升级协议

BASE64 不发松请求(好用但是要慎用,加大了文件体积)  

6、图片懒加载 / 预加载

网络层优化

CRP浏览器关键节点优化 (critical rendering path)

webpack层优化

安全优化

代码层优化 闭包

相关文章
|
23天前
|
机器学习/深度学习 人工智能 前端开发
【人工智能】利用TensorFlow.js在浏览器中实现一个基本的情感分析系统
使用TensorFlow.js在浏览器中进行情感分析是一个非常实用的应用场景。TensorFlow.js 是一个用于在JavaScript环境中训练和部署机器学习模型的库,使得开发者能够在客户端直接运行复杂的机器学习任务。对于情感分析,我们可以使用预先训练好的模型来识别文本中的积极、消极或中性情感。
47 4
【人工智能】利用TensorFlow.js在浏览器中实现一个基本的情感分析系统
|
14天前
|
编解码 JavaScript 前端开发
JS逆向浏览器脱环境专题:事件学习和编写、DOM和BOM结构、指纹验证排查、代理自吐环境通杀环境检测、脱环境框架、脱环境插件解决
JS逆向浏览器脱环境专题:事件学习和编写、DOM和BOM结构、指纹验证排查、代理自吐环境通杀环境检测、脱环境框架、脱环境插件解决
37 1
|
17天前
|
JavaScript 前端开发 算法
js 内存回收机制
【8月更文挑战第23天】js 内存回收机制
29 3
|
17天前
|
存储 JavaScript 前端开发
学习JavaScript 内存机制
【8月更文挑战第23天】学习JavaScript 内存机制
16 3
|
17天前
|
JavaScript
js渲染乘法表
js渲染乘法表
22 1
|
20天前
|
JavaScript 算法 前端开发
"揭秘Vue.js的高效渲染秘诀:深度解析Diff算法如何让前端开发快人一步"
【8月更文挑战第20天】Vue.js是一款备受欢迎的前端框架,以其声明式的响应式数据绑定和组件化开发著称。在Vue中,Diff算法是核心之一,它高效计算虚拟DOM更新时所需的最小实际DOM变更,确保界面快速准确更新。算法通过比较新旧虚拟DOM树的同层级节点,递归检查子节点,并利用`key`属性优化列表更新。虽然存在局限性,如难以处理跨层级节点移动,但Diff算法仍是Vue高效更新机制的关键,帮助开发者构建高性能Web应用。
35 1
|
8天前
|
JavaScript 中间件 开发者
深入浅出Node.js中间件机制
【8月更文挑战第31天】本文将带你领略Node.js中间件的奥秘,通过直观的案例分析,揭示其背后的设计哲学。你将学会如何运用中间件构建强大而灵活的后端应用,以及在面对复杂业务逻辑时如何保持代码的清晰与高效。
|
8天前
|
API UED 开发者
如何在Uno Platform中轻松实现流畅动画效果——从基础到优化,全方位打造用户友好的动态交互体验!
【8月更文挑战第31天】在开发跨平台应用时,确保用户界面流畅且具吸引力至关重要。Uno Platform 作为多端统一的开发框架,不仅支持跨系统应用开发,还能通过优化实现流畅动画,增强用户体验。本文探讨了Uno Platform中实现流畅动画的多个方面,包括动画基础、性能优化、实践技巧及问题排查,帮助开发者掌握具体优化策略,提升应用质量与用户满意度。通过合理利用故事板、减少布局复杂性、使用硬件加速等技术,结合异步方法与预设缓存技巧,开发者能够创建美观且流畅的动画效果。
31 0
|
8天前
|
设计模式 JavaScript 中间件
深入浅出Node.js中间件机制
【8月更文挑战第31天】在Node.js的世界里,中间件如同魔法般存在,它让复杂的请求处理变得井然有序。本文将带你领略中间件的奥秘,从原理到实战,一步步揭开它的神秘面纱。你将学会如何运用中间件来构建强大而灵活的后端应用,就像拼乐高一样有趣。
|
10天前
|
JavaScript 前端开发 UED
服务器端渲染新浪潮:用Vue.js和Nuxt.js构建高性能Web应用
【8月更文挑战第30天】在现代Web开发中,提升应用性能和SEO友好性是前端开发者面临的挑战。服务器端渲染(SSR)能加快页面加载速度并改善搜索引擎优化。Vue.js结合Nuxt.js提供了一个高效框架来创建SSR应用。通过安装`create-nuxt-app`,可以轻松创建新的Nuxt.js项目,并利用其自动路由功能简化页面管理。Nuxt.js默认采用SSR模式,并支持通过`asyncData`方法预取数据,同时提供了静态站点生成和服务器端渲染的部署选项,显著提升用户体验。
29 0