普通视图

发现新文章,点击刷新页面。
昨天 — 2025年6月9日首页

为什么我们用了 Vite 还是构建慢?——真正的优化在这几步

作者 ErpanOmer
2025年6月9日 09:57

Vite 凭借其基于原生 ESM 的开发体验和极速冷启动,被誉为“现代前端构建的终极解决方案”。然而,许多开发者在将老项目迁移到 Vite 或新项目上马后,仍会面临以下现实:

  • 冷启动是快了,但热更新开始卡顿
  • 构建产物还是几十秒甚至上百秒
  • vite build 一跑,CPU 风扇飞起,内存飙升

这不禁让人怀疑:Vite 真的快吗?我是不是哪里搞错了?

答案是:Vite 本身没错,但你可能忽略了 “构建速度的决定性因素” 并不只是工具,而是你整个工程的结构与使用方式。


第一步:认清 Vite 的本质——开发快 ≠ 构建快

Vite 的核心价值之一是:

开发时基于原生 ESM 模块按需加载,只构建你当前用到的模块。

所以你在开发时能享受到:

  • 秒级冷启动
  • 快速热更新(HMR)

但是一旦 vite build,它会切换到底层的 Rollup 作为打包器。这时候:

  • 所有依赖和模块都会被打包(并非按需加载)
  • 你项目里隐藏的复杂性就无所遁形了

如果你项目构建慢,问题通常出在这几个地方 👇


第二步:从源码开始拆解瓶颈

1. 依赖体积巨大,tree-shaking 无效

常见现象:

  • 引入了 lodash,但只用了一个 cloneDeep,结果全库都进来了
  • 三方组件库未按需引入
  • import 了整个 moment/antd/dayjs/chart.js 等大块头

建议:

  • 替换为按需引入版本(例如 lodash-esdayjs
  • 配置 vite-plugin-impunplugin-vue-components 等插件实现自动按需加载
  • 使用 esbuild 插件提前预构建大型依赖

2. 依赖过多,预构建时间拉长

Vite 会在第一次启动时对第三方依赖进行 预构建,但如果:

  • node_modules 太大
  • dependenciesdevDependencies 没分清
  • 重复安装多个版本的库

则预构建变成噩梦。

建议:

  • 使用 optimizeDeps.exclude 排除不必要的模块
  • 确保依赖升级一致,避免 lodash@3lodash@4 共存
  • 使用 pnpmyarn workspace 做 hoist 减少重复依赖

3. 大量动态导入 / 动态路由

const page = import(`./pages/${name}.vue`)

这类动态导入虽然在开发阶段无感,但在打包时会造成:

  • 无法静态分析,打包粒度变粗
  • chunk 切割不理想

建议:

  • 避免使用过多复杂路径的 import()
  • 显式声明 chunk 名称:
const page = import(/* webpackChunkName: "page-[request]" */ `./pages/${name}.vue`)
  • 使用插件 vite-plugin-pages 提前生成路由,提升打包可控性

4. 打包时文件过多或模块层级过深

Vite 最怕你这样:

  • 单页项目动辄几千个 .vue 文件
  • 组件层级 6 层嵌套,导入链冗长

这样会造成:

  • Rollup 构建图构造时间大大增加
  • chunk 分析变慢

建议:

  • 控制模块划分粒度
  • 重构庞大的组件结构
  • 使用 vite-plugin-inspectwhy-so-slow 工具分析依赖链路

第三步:构建优化技巧全集

✅ 开启 Rollup 缓存(生产构建)

export default defineConfig({
  build: {
    cacheDir: 'node_modules/.vite_cache',
  },
})

✅ 使用 esbuild 插件压缩而不是 Terser

build: {
  minify: 'esbuild', // 默认为 terser,esbuild 更快
}

✅ 压缩大型依赖提前剥离(optimizeDeps + manualChunks)

optimizeDeps: {
  include: ['axios', 'lodash-es'],
}
rollupOptions: {
  output: {
    manualChunks: {
      vendor: ['vue', 'vue-router'],
    },
  },
}

✅ 设置 SSR-friendly 构建配置

如果你的项目未来有 Nuxt / SSR 的诉求:

ssr: {
  noExternal: ['your-lib'], // 避免库在服务端打包出错
}

第四步:缓存、预构建、硬盘读写也重要

  • .vite 缓存目录删除后构建速度会变慢一次
  • node_modules 删除后会触发重新 optimizeDeps
  • 高并发文件读取(尤其在 HDD 硬盘上)会导致构建性能极差

建议:

  • 使用 SSD 做本地开发盘
  • 保留 .vite 缓存目录
  • 在 CI/CD 中加缓存提升构建速度

Vite 是好工具,但别用错姿势

Vite 真的快,但也真的依赖你的项目结构合理性。如果你像用 Webpack 一样对待 Vite,把所有依赖全部一锅端地丢进来,还觉得构建变慢了,那问题不在工具,而在使用方式。

所以,如果你真的想优化构建速度:

  • 别急着 blame Vite
  • 先看看自己是不是喂了它太多“垃圾”代码
昨天以前首页

你以为的 Tailwind 并不高效,看看这些使用误区

作者 ErpanOmer
2025年6月6日 13:54

“Tailwind 写得越多,越觉得混乱”“组件样式重复一堆”“设计师完全看不懂这坨 className”…

这些反馈你是否也听说过?

Tailwind CSS 被誉为“实用优先的 CSS 框架”,然而在实际项目中,很多团队用了 Tailwind,效率却不升反降
不是因为 Tailwind 本身不行,而是——你可能正踩在这些使用误区上

本文将围绕 6 个常见误区,逐一剖析:

  1. 把 Tailwind 当成原子 CSS 的“组合器”
  2. 滥用 @apply 和组件式提取
  3. 不配置 Design Token,直接用默认色板
  4. 缺少抽象语义类的规范
  5. 无视团队协作中的语义歧义
  6. 不懂插件生态和可配置性,白白造轮子

🎯 误区一:把 Tailwind 当成“低配版 SCSS”来写

有些团队习惯了 BEM 或 SCSS 的写法,迁移 Tailwind 后陷入一个坑:

“我只会把 .title {} 改写成 <div class="title">,然后 .title@apply text-lg font-bold 合成”

结果就是:

  • 每个组件依然维护 .xxx { @apply ... }
  • 每个 class 被重复“硬编码”,没有复用价值
  • 真正的 Tailwind 优势(原子化组合)彻底失效

这其实是把 Tailwind 当成 SCSS 的语法糖在用,不仅没提高效率,反而多了一层“拼贴工”。

建议:

  • 组件中直接使用原子类组合,不要回退为传统 class 命名法
  • 通过抽象语义 class(例如 btn-primary)统一复杂样式,避免 @apply 滥用

🔥 误区二:滥用 @apply,导致样式复用变得更难维护

Tailwind 支持 @apply,但很多人误解它是“推荐方式”。结果就是:

.btn {
  @apply px-4 py-2 rounded-md bg-blue-500 text-white;
}
.btn-secondary {
  @apply btn bg-gray-500;
}

这本质上已经偏离了 Tailwind 的精神。问题在于:

  • @apply 语义缺失,又回到了传统 CSS 的“找 class 写样式”流程
  • 一旦 .btn 修改,影响范围不可控(链式引用)
  • 无法动态响应状态(例如 hover:bg-blue-600dark:bg-blue-400

正确方式:

  • 抽象出来的 class 应该直接写在 class="" 中,比如 class="btn btn-primary"
  • 配合 UnoCSS 或 Tailwind plugin,使用 语义化原子类 实现动态组合(见下文)

🎨 误区三:直接使用 Tailwind 默认色板,导致主题难以统一

许多初学者习惯直接写:

<div class="bg-blue-500 text-gray-800">按钮</div>

看起来没毛病,但:

  • “blue-500” 具体代表什么品牌色?设计稿里用的是 #378AFF 你知道吗?
  • 一旦品牌换主色,100 个组件都要人工搜索替换?
  • 多人项目中,“每个人对 text-sm 的认知都不同”

这不是视觉认知问题,而是你没有建立设计 token 体系

推荐配置方式(tailwind.config.ts):

theme: {
  colors: {
    brand: {
      DEFAULT: '#378AFF',
      dark: '#2563EB',
      light: '#93C5FD'
    }
  },
  fontSize: {
    base: '16px',
    sm: '14px',
    lg: '18px'
  }
}

然后组件中统一用 bg-brand text-sm,做到真正的“设计系统驱动”。


🧱 误区四:class 混乱、语义缺失,导致组件难复用

很多初级 Tailwind 项目里的组件看起来像这样:

<div class="px-4 py-2 rounded-md bg-blue-500 text-white text-sm shadow-md hover:bg-blue-600">
  提交
</div>

逻辑没问题,但:

  • 这个组件到底是按钮?标签?还是 Toast?
  • 想要复用时只能 copy-paste,一改就坏
  • 业务迭代时,10 个类似组件居然样式不同(改过一点 padding、换了一个 shadow)

Tailwind 推荐语义化命名 + utility 组合方式,例如:

<button class="btn btn-primary">提交</button>

然后在 tailwind.config.ts 里添加:

plugins: [
  require('@tailwindcss/forms'),
  function ({ addComponents }) {
    addComponents({
      '.btn': {
        @apply px-4 py-2 rounded-md text-white text-sm;
      },
      '.btn-primary': {
        @apply bg-brand hover:bg-brand-dark;
      }
    })
  }
]

不仅可复用,还能集中管理样式变更。


🧠 误区五:团队协作不统一,样式风格“各写各的”

当团队使用 Tailwind 却没有协作规范时,经常出现以下现象:

  • A 开发写 text-sm, B 写 text-xs,页面出现 4 种字号
  • 有的写 rounded, 有的写 rounded-md,风格混乱
  • 多人维护组件时,一改颜色就影响全局,因为写死在组件里

Tailwind 鼓励显式地“写出来”,但这不代表可以无限自由拼接

解决方式:

  • 建立 token 规范:颜色、字体、尺寸统一配置
  • 使用 @shadcn/ui 作为组件层抽象(推荐组合 Tailwind + Radix + shadcn)
  • 使用 ESLint 插件强制校验 Tailwind class 的顺序和规范(如 eslint-plugin-tailwindcss

⚙️ 误区六:没用插件、没开 JIT、错失生态红利

很多团队用了 Tailwind,但配置还停留在最基本阶段:

  • 没开 JIT 模式(Just In Time 构建)
  • 没启用 dark mode(Tailwind 支持类选择器控制 dark 样式)
  • 没接入 tailwind-variants / clsx / cva 等原子类组合库
  • 对 UnoCSS 完全不了解(其实比 Tailwind 更自由,兼容性好)

Tailwind 的能力远不止于“写几个 class”——它已经是一个完整的样式编程语言了

举个例子:

你可以使用 tailwind-variants 这样写组件样式组合:

const button = tv({
  base: 'inline-flex items-center justify-center font-medium',
  variants: {
    intent: {
      primary: 'bg-brand text-white hover:bg-brand-dark',
      secondary: 'bg-gray-100 text-gray-800 hover:bg-gray-200'
    },
    size: {
      sm: 'text-sm py-1 px-2',
      md: 'text-base py-2 px-4'
    }
  }
})

然后在组件中使用:

<button className={button({ intent: 'primary', size: 'md' })}>按钮</button>

写起来干净,组合灵活,完全符合设计系统的思想。


✅ 正确的 Tailwind 使用思维:构建语义原子设计系统

Tailwind 的真正优势不在于“快”,而在于:

构建一套“视觉样式与设计语言一致”的原子组件体系。

好的实践是:

  • 所有颜色、字号、间距抽象为 Design Token
  • 所有组件样式统一在配置文件或 plugin 中抽象复用
  • 使用 shadcn/ui + tailwind-variants + clsx 构建语义组件库
  • 使用 prettier-plugin-tailwindcss + ESLint 强化规范
  • 页面代码中只使用 最小必要 class + 语义化 class
  • 样式变化驱动来自配置层变动,而非组件层硬改

🧩 最后

Tailwind 并不是魔法,它只是一个极致实用主义的 CSS 工具包。

真正的工程实践中,Tailwind 的效率取决于你是否理解并规避以下误区:

误区 正确思路
把 Tailwind 当 SCSS 用 原子化组合 + 抽象语义 class
滥用 @apply 提炼插件组件 + 原子类组合方式
使用默认色板无品牌感 建立 Design Token 体系
样式写死、组件无法复用 抽象组件语义 class + 统一配置管理
多人协作风格混乱 使用 lint、格式化插件强制规范
没使用生态工具,错失红利 熟悉 tailwind-variants / UnoCSS 等替代方案
❌
❌