什么是vite系列目录:
(二)什么是Vite——Vite 和 Webpack 区别(冷启动)-CSDN博客
(三)什么是Vite——Vite 主体流程(运行npm run dev后发生了什么?)-CSDN博客
未完待续。。。
- |
- -CHANGELOG.md
- |-LICENSE.md
- |-README.md
- |-bin
- | |-openChrome.applescript
- | |-vite.js
- |-client.d.ts
- |-package.json
- |-rollup.config.js #打包配置文件
- |-scripts
- | |-patchTypes.js
- |-src
- | |-client #客户端
- | | |-client.ts
- | | |-env.ts
- | | |-overlay.ts
- | | |-tsconfig.json
- | |-node #服务端
- | | |-build.ts
- | | |-cli.ts #命令入口文件
- | | |-config.ts
- | | |-constants.ts #常量
- | | |-importGlob.ts
- | | |-index.ts
- | | |-logger.ts
- | | |-optimizer
- | | | |-esbuildDepPlugin.ts
- | | | |-index.ts
- | | | |-registerMissing.ts
- | | | |-scan.ts
- | | |-plugin.ts #rollup 插件
- | | |-plugins #插件相关文件
- | | | |-asset.ts
- | | | |-clientInjections.ts
- | | | |-css.ts
- | | | |-esbuild.ts
- | | | |-html.ts
- | | | |-index.ts
- | | | |-...
- | | |-preview.ts
- | | |-server
- | | | |-hmr.ts #热更新
- | | | |-http.ts
- | | | |-index.ts
- | | | |-middlewares #中间件
- | | | | |-...
- | | | |-moduleGraph.ts #模块间关系组装(树形)
- | | | |-openBrowser.ts #打开浏览器
- | | | |-pluginContainer.ts
- | | | |-send.ts
- | | | |-sourcemap.ts
- | | | |-transformRequest.ts
- | | | |-ws.ts
- | | |-ssr
- | | | |-__tests__
- | | | | |-ssrTransform.spec.ts
- | | | |-ssrExternal.ts
- | | | |-ssrManifestPlugin.ts
- | | | |-ssrModuleLoader.ts
- | | | |-ssrStacktrace.ts
- | | | |-ssrTransform.ts
- | | |-tsconfig.json
- | | |-utils.ts
- |-tsconfig.base.json
- |-types
- | |-...
(初识 vite 原理,vite 是如何启动项目的)(源码调试:下一代前端构建工具 - Vite 2.x 源码级分析)
vite命令是在哪里注册的呢,在 node_modules/vite/package.json 中查看bin字段。"bin" 字段的作用是能让我们在命令窗口全局输入命令执行。vite启动的时候,会执行 bin 目录下的 vite.js 文件,在这个文件里面我们会看到获取了当前的电脑的 绝对路径 ,如果不包含 node_module 路径,就需要引入 source-map-support 这个包来处理,如果是相对路径则会进行路径补全,当然在 dev 环境下都会有 node_module 路径,在 prod 环境下, vite 会使用 rollup 来进行打包。

示例项目下:package-lock.json

再示例项目下bin文件下看到三个vite文件,分别是针对unity,windows,跨平台三个方向链接对应的环境变量:

打开项目中的vite.js看到,其中主要运行的是从入口文件 cli.js。
vite.js 中的核心内容就是执行了 start 方法,动态引入了 ../dist/node/cli.js ,这个地址是打包后的地址,在 vite 源码中,脚本地址在 packages/vite/src/node/cli.ts 。

这里的/dist/node/cli.js是打包后的文件,可能有点长,可以配合vite打包前的源码一起阅读:

在 /dist/node/cli.js 中,首先使用 cac 命令工具处理用户的输入。

然后命令执行的是 createServer,这个 createServer 从 ./chunks/dep-df561101.js 引入,这里也是打包后的代码,比较长。
cli.ts 的核心功能是解析命令行参数并启动本地项目,解析命令行参数通过 cac 库,这里我们主要看启动本地项目的命令。在 cac 库中,通过 command 定义基础命令,通过 alias 方法定于命令别名,通过 option 方法定义命令行参数,最后通过 action 方法执行具体的操作。在 action 方法中,最核心的部分就是引入了 createServer 方法,通过 listen 启动本地 server。

同时在 cli.js 文件中,可以看到四个命令对应了 4 个核心的文件&方法:
文件路径:./server/index.ts;
主要方法:createServer;
主要功能:项目的本地开发命令,基于 httpServer 启动服务,Vite 通过对请求路径的劫持获取资源的内容返回给浏览器,服务端将文件路径进行了重写。例如:
项目源码如下:
- import { createApp } from 'vue';
- import App from './index.vue';
经服务端重写后,node_modules 文件夹下的三方包代码路径也会被拼接完整。
- import __vite__cjsImport0_vue from "/node_modules/.vite/vue.js?v=ed69bae0";
- const createApp = __vite__cjsImport0_vue["createApp"];
- import App from '/src/pages/back-sky/index.vue';
主要方法:build;
主要功能:使用 rollup 打包编译
文件路径:./optimizer/index.ts;
主要方法:optimizeDeps;
主要功能:主要针对第三方包,Vite 在执行 runOptimize 的时候中会使用 rollup 对三方包重新编译,将编译成符合 esm 模块规范的新的包放入 node_modules 下的 .vite 中,然后配合 resolver 对三方包的导入进行处理:使用编译后的包内容代替原来包的内容,这样就解决了 Vite 中不能使用 cjs 包的问题。
下面是 .vite 文件夹中的 _metadata.json 文件,它在预编译的过程中生成,罗列了所有被预编译完成的文件及其路径。例如:
- {
- "hash": "1848098d",
- "browserHash": "f4266ce6",
- "optimized": {
- "./src/lib/index.js": {
- "src": "../../../src/lib/index.js",
- "file": "___src_lib_index__js.js",
- "fileHash": "58a83bbc",
- "needsInterop": false
- },
- "lodash-es": {
- "src": "../../lodash-es/lodash.js",
- "file": "lodash-es.js",
- "fileHash": "915b5f31",
- "needsInterop": false
- },
- "vue": {
- "src": "../../vue/dist/vue.runtime.esm-bundler.js",
- "file": "vue.js",
- "fileHash": "c91074d2",
- "needsInterop": false
- }
- },
- "chunks": {}
- }
文件路径:./preview/index.ts;
主要方法:preview;
主要功能:本地预览构建产物。不要将其用作生产服务器,因为它不是为此而设计的。
我们找到了前面大概执行的顺序后,这里回到源码,在 packages/vite/src/node/server/index.ts 里面找到 createServer:
- export async function createServer(
- inlineConfig: InlineConfig = {}
- ): Promise<ViteDevServer> {
- // Vite 配置整合
- // resolveConfig 方法解析 vite 核心配置,包括来自命令行、vite.config 文件的配置参数
- const config = await resolveConfig(inlineConfig, 'serve', 'development')
- const root = config.root
- const serverConfig = config.server
-
- // 创建http服务,根目录的 index.html 就是服务器的入口
- const httpServer = await resolveHttpServer(serverConfig, middlewares, httpsOptions)
-
- // 创建ws服务,主要是用于实现热更新(HMR),当代码发生变化时,服务器通过 WebSocket 向客户端发送更新通知
- const ws = createWebSocketServer(httpServer, config, httpsOptions)
-
- // 创建watcher,设置代码文件监听,实时地响应文件的增删改操作,也是用于实现热更新功能
- const watcher = chokidar.watch(path.resolve(root), {
- ignored: [
- '**/node_modules/**',
- '**/.git/**',
- ...(Array.isArray(ignored) ? ignored : [ignored])
- ],
- ...watchOptions
- }) as FSWatcher
-
- // 创建server对象
- const server: ViteDevServer = {
- config, // 配置属性
- middlewares, // 中间件
- httpServer, // HTTP server 实例
- watcher, // chokidar 文件监听实例
- pluginContainer: container, // 插件容器
- ws, // WebSocket 实例
- moduleGraph, // 模块依赖图
- listen,
- ...
- }
-
- // 文件监听变动,websocket向前端通信
- watcher.on('change', async (file) => {
- ...
- handleHMRUpdate()
- })
-
- // 服务 middleware,通过 use 方法添加启动项目阶段需要的中间件
- middlewares.use(...)
-
- // optimize: 预构建
- await initDepsOptimizer(config, server)
-
- // 监听端口,启动服务
- httpServer.listen = (async (port, ...args) => { ... })
-
- return server
- }
可以看到 createServer 做了很多事情:

最后总结一下,在开发过程中,vite 启动命令 npm run dev 执行后,实际执行的是 node_module/.bin 目录下的 vite 脚本,在解析命令行参数之后,通过执行 createServer.listen 方法启动 vite 。

下图是 Vite 在开发环境运行时加载文件的主体流程。

具体如下:
在项目根目录下有一个index.html文件,在其script标签设置type='module':

浏览器会向服务器发起一个GET请求,拿到main.js文件:

- http://localhost:3000/src/main.js请求main.js文件:
-
- // /src/main.js:
- import { createApp } from 'vue'
- import App from './App.vue'
- createApp(App).mount('#app')
