背景

随着vue3的到来,vite开始被各大vue3组件库使用,公司开始一个新项目,准备尝试用vite试一波。

问题发现

当把公司新项目移植到vite后,启动非常快,但发现页渲染时间慢了很多

可以看到页面的首屏加载时间是3.34s,页面的渲染完时间是3.37s,下载总大小是8.6MB,发送了119个请求

在看看webpack的渲染时间:

可以看到页面的首屏加载时间是1.05s,页面的渲染完时间是1.21s,下载总大小是3.6MB,发送了32个请求

vite的首屏加载时间是webpack的三倍,下载文件大小是webpack的两倍,请求数量也是三倍。vite开发环境下,模块以原生 esm 的形式被浏览器加载,也就说模块的加载是用es6原生的模块加载机制,没有对代码进行打包压缩处理,所以服务启动很快。那带来的问题就是下载的js文件没有处理的过的源码,那文件大小自然要比webpack处理过的js文件大很多。

因此初步判断因为这个原因导致首屏加载时间相差这么多。得出结论vite是牺牲了页面首次加载时间来达到启动时间快的目的。

峰回路转

于是我去网上寻找有没有好的解决方案,在vite的issue中找到类似的问题:

尤大大也回答了这个问题

这个问题有两个细节:

  1. 项目启动后浏览器第一次加载才会慢。
  2. 这个慢是因为加载less的源码,按需编译中加载时间其实是在less的编译上。

看看vite项目第二次加载

可以看到页面的首屏加载时间是1.04s,页面的渲染完时间是1.09s,下载总大小是8.6MB,发送了120个请求

页面的渲染时间和webpack项目的渲染时间差不多,esm 的形式被浏览器加载和请求数量对页面的渲染时间的影响不是主要的,也证实项目启动后浏览器第一次加载多出的时间主要是在编译上。

尤大大在知乎回答中提到:“webpack 的打包模式在项目本身源码模块数量极大 (>1000) 的情况下还是有一点优势的,因为浏览器在处理这个级别的并发请求上会产生阻塞(但通常来说如果你一个路由下模块数到这个级别说明你代码分割/按需加载没做好)。”意思是esm 的形式被浏览器加载和请求数量对页面的渲染时间的影响不是主要的。

分析

浏览器第一次加载时间比较长的文件时间分部:

waterfall性能检测详解详解:

  • Queueing 是排队的意思
  • Stalled 是阻塞 请求访问该URL的主机是有并发和连接数限制的,必须要等之前的执行才能执行之后的,这段时间的耗时
  • DNS Lookup 是指域名解析所耗时间
  • Initial connection 初始化连接时间,这里一般是TCP 3次连接握手时间
  • SSL https特有,是一种协议
  • Request sent 发送请求所消耗的时间
  • Waiting 等待响应时间,这里一般是最耗时的
  • Content Download 下载内容所需要消耗的时间

加载index.less的时间主要在Waiting阶段,Content Download的时间占比很少。等待响应时间应该就是vite在编译的时间。

我的项目也引用了组件库的less源码,改为引用组件库的css看下渲染时间如何

可以看到页面的首屏加载时间是1.13s,页面的渲染完时间是1.16s,下载总大小是8.4MB,发送了120个请求

渲染速度基本和webpack差不多。

结论

通过上面分析可以确定vite开发模式启动,页面加载慢的原因是less编译导致。

这里反思下,由于对调试工具waterfall性能检测不熟悉导致开始错误的结论,走了一下弯路。有时候我们通过理论推导出来的结论未必正确的,需要实际数据去证实但更需要去证伪。就像吴军老师说的:

“我们每次给出的结论,都存在一种证伪的可能性,我们可以设计出一种方法、一个实验,或者一个场景,去看看我们的结论是不是还是成立,如果成立,我们就维持现有的结论;如果不成立,我们就修改现有结论。这就叫做证伪。

科学是一种做事情的方法,而不是特定的结论。采用科学方法得到的结论,相对来讲,比较经得起时间和现实世界的检验。更重要的是,用科学的方法求知,我们能得到可累加的进步,因为每一次验证的过程都是可以重复,后面的工作也可以基于前面的工作展开。”这句话牢记心中。

vite首次启动加载慢的更多相关文章

  1. AngularJS进阶(三十九)基于项目实战解析ng启动加载过程

    基于项目实战解析ng启动加载过程 前言 在AngularJS项目开发过程中,自己将遇到的问题进行了整理.回过头来总结一下angular的启动过程. 下面以实际项目为例进行简要讲解. 1.载入ng库 2 ...

  2. Microsoft Corporation 去掉 windows 修改 启动加载 版权

    windows 修改 开机界面 boot启动界面 windows 修改 启动加载 版权 windows 系统如何修改开机画面的版权文字“Microsoft Corporation ... ◎Micro ...

  3. SpringBoot启动加载类ApplicationRunner

    SpringBoot启动加载类ApplicationRunner 有时希望项目在启动的时候加载一些系统参数,就要用到ApplicationRunner ApplicationRunner是一个接口,我 ...

  4. SpringBoot(七)-- 启动加载数据

    一.场景 实际应用中,我们会有在项目服务启动的时候就去加载一些数据或做一些事情这样的需求.为了解决这样的问题,spring Boot 为我们提供了一个方法,通过实现接口 CommandLineRunn ...

  5. JavaWeb 服务启动时,在后台启动加载一个线程

    JavaWeb 服务启动时,在后台启动加载一个线程. 目前,我所掌握的一共有两种方法,第一种是监听(Listener),第二种是配置随项目启动而启动的Servlet. 下面对这两种方法做一简单的介绍, ...

  6. redis启动加载过程、数据持久化

    背景 公司一年的部分业务数据放在redis服务器上,但数据量比较大,单纯的string类型数据一年就将近32G,而且是经过压缩后的. 所以我在想能否通过获取string数据的时间改为保存list数据类 ...

  7. spring boot启动加载项CommandLineRunner

    spring boot启动加载项CommandLineRunner 在使用SpringBoot构建项目时,我们通常有一些预先数据的加载.那么SpringBoot提供了一个简单的方式来实现–Comman ...

  8. Tomcat源码分析三:Tomcat启动加载过程(一)的源码解析

    Tomcat启动加载过程(一)的源码解析 今天,我将分享用源码的方式讲解Tomcat启动的加载过程,关于Tomcat的架构请参阅<Tomcat源码分析二:先看看Tomcat的整体架构>一文 ...

  9. Windows机器配置启动加载器的高级选项后,机器出现蓝屏,无法RDP

    问题描述: 虚拟机重启后出现蓝屏,需要排查原因 蓝屏界面如下: 排查结果: 1.Console发现机器停留在上述蓝屏界面,按F8后,机器可以正常启动并成功连接:但通过console再次重启后,又会进入 ...

随机推荐

  1. Ant高级-path和fileset

    一 <path/> 和 <classpath/> 你可以用":"和";"作为分隔符,指定类似PATH和CLASSPATH的引用.Ant会 ...

  2. Map 综述(一):彻头彻尾理解 HashMap

    转载自:https://blog.csdn.net/justloveyou_/article/details/62893086 摘要: HashMap是Map族中最为常用的一种,也是 Java Col ...

  3. [leetcode]1109. 航班预订统计(击败100%用户算法-差分数组的详解)

    执行用时2ms,击败100%用户 内存消耗52.1MB,击败91%用户 这也是我第一次用差分数组,之前从来没有碰到过,利用差分数组就是利用了差分数组在某一区间内同时加减情况,只会改变最左边和最右边+1 ...

  4. centos7 修改时间

    2021-07-28 查看日志时发现系统时间不正确,故修改时间 # 查看当前系统时间 date # 修改当前系统时间 date -s "2021-7-28 17:03:00" # ...

  5. vue-class和style样式绑定

    前言 操作元素的 class 样式列表和 style 内联样式为数据绑定是前端开发中一个常见的需求,这些样式都属于元素的属性 attribute ,因此我们可以通过 v-bind 来动态绑定元素的样式 ...

  6. .NetCore 项目在服务器打包失败解决

    错误描述:NuGet警告 NU3037 NU3028 原因:Nuget无法访问到json所在的网络 2021年1月31日更新:更好的方法 把自动生成的Dockerfile内的AS build 替换成官 ...

  7. 基于Appium,封装自己的常用方法

    Appium算是老牌移动端App自动化测试工具了,在使用它的过程中,使用者经常会根据个人习惯,把较常用的方法封装在一起,方便调用.以下是我的封装,希望对你有启发. from typing import ...

  8. K8S集群架构的组件组成

    1.Master--主控节点 (1)apiserver:集群统一入口,以restful的方式,交给etcd存储 (2)scheduler:节点调度,选择node节点应用部署 (3)controller ...

  9. adb 常用命令大全(6)- 模拟按键输入

    语法格式 input [<source>] <command> [<arg>...] 物理键 # 电源键 adb shell input keyevent 26 # ...

  10. 密码学系列之:bcrypt加密算法详解

    目录 简介 bcrypt的工作原理 bcrypt算法实现 bcrypt hash的结构 hash的历史 简介 今天要给大家介绍的一种加密算法叫做bcrypt, bcrypt是由Niels Provos ...