Part3-5-4 Vite

Posted 沿着路走到底

tags:

篇首语:本文由小常识网(cha138.com)小编为大家整理,主要介绍了Part3-5-4 Vite相关的知识,希望对你有一定的参考价值。

Vite

Vite 是一个面向现代浏览器的一个更轻、更快的 Web 应用开发工具。

它基于 ECMAScript 标准原生模块系统(ES Modules)实现。

Vite 项目依赖

Vite

@vue/compiler-sfc  // 用来编译 .vue 结尾的单文件组件。

基础使用

vite serve

vite buiild

HMR

Vite HMR

立即编译当前所修改的文件

Webpack HMR

会自动以这个文件为入口重写 build 一次,所有的涉及到的依赖也都会被加载一遍。

Bulid

vite build

        Rollup

        Dynamic import  // 动态导入

                Polyfill

使用 webpack 打包的二个原因

浏览器环境并不支持模块化

零散的模块文件会产生大量的 HTTP 请求

开箱即用

TypeScript - 内置支持

less / sass / stylus / postcss  - 内置支持 (需要单独安装)

JSX

Web Assembly

Vite 特性

快速冷启动

模块热更新

按需编译

开箱即用

Vite 核心功能

开启一个静态 Web 服务器

编译单文件组件

        拦截浏览器不识别的模块,并处理

HMR

Vite-cli

#!/usr/bin/env node
const path = require('path')
const { Readable } = require('stream')
const Koa = require('koa')
const send = require('koa-send')
const compilerSFC = require('@vue/compiler-sfc')

const app = new Koa()

const streamToString = stream => new Promise((resolve, reject) => {
  const chunks = []
  stream.on('data', chunk => chunks.push(chunk))
  stream.on('end', () => resolve(Buffer.concat(chunks).toString('utf-8')))
  stream.on('error', reject)
})

const stringToStream = text => {
  const stream = new Readable()
  stream.push(text)
  stream.push(null)
  return stream
}

// 3. 加载第三方模块
app.use(async (ctx, next) => {
  // ctx.path --> /@modules/vue
  if (ctx.path.startsWith('/@modules/')) {
    const moduleName = ctx.path.substr(10)
    const pkgPath = path.join(process.cwd(), 'node_modules', moduleName, 'package.json')
    const pkg = require(pkgPath)
    ctx.path = path.join('/node_modules', moduleName, pkg.module)
  }
  await next()
})

// 1. 静态文件服务器
app.use(async (ctx, next) => {
  await send(ctx, ctx.path, { root: process.cwd(), index: 'index.html' })
  await next()
})

// 4. 处理单文件组件
app.use(async (ctx, next) => {
  if (ctx.path.endsWith('.vue')) {
    const contents = await streamToString(ctx.body)
    const { descriptor } = compilerSFC.parse(contents)
    let code
    if (!ctx.query.type) {
      code = descriptor.script.content
      // console.log(code)
      code = code.replace(/export\\s+default\\s+/g, 'const __script = ')
      code += `
      import { render as __render } from "${ctx.path}?type=template"
      __script.render = __render
      export default __script
      `
    } else if (ctx.query.type === 'template') {
      const templateRender = compilerSFC.compileTemplate({ source: descriptor.template.content })
      code = templateRender.code
    }
    ctx.type = 'application/javascript'
    ctx.body = stringToStream(code)
  }
  await next()
})

// 2. 修改第三方模块的路径
app.use(async (ctx, next) => {
  if (ctx.type === 'application/javascript') {
    const contents = await streamToString(ctx.body)
    // import vue from 'vue'
    // import App from './App.vue'
    ctx.body = contents
      .replace(/(from\\s+['"])(?![\\.\\/])/g, '$1/@modules/')
      .replace(/process\\.env\\.NODE_ENV/g, '"development"')
  }
})

app.listen(3000)
console.log('Server running @ http://localhost:3000')

1

以上是关于Part3-5-4 Vite的主要内容,如果未能解决你的问题,请参考以下文章

来了来了,最新vite源码分析,vite到底为什么比webpack快

浅析Vite2.0-依赖预打包

vite识别.vue文件

使用 Vite 做后端

vite 开发 Cesium 程序最佳配置实践

如何通过 VITE 在 Data() Vue 3 组件中使用 JSX/HTML 代码