webpack5 + vue3 从零配置项目
前言
虽然在实际项目当中很少会从 0 到 1 配置一个项目,毕竟很多重复工作是没有必要的,脚手架将这些重复性的工作进行了整合,方便开发者使用。也正因如此,导致部分开发者过于依赖脚手架,却不清楚其内部的实现流程,因此通过从 0 到 1 去配置和搭建项目可以更好的理解开发中使用的脚手架都帮我们做了哪些事情。
准备工作
创建目录结构
这里直接使用 npm init vite@latest
命令生成 vue3
最新的目录结构:
当然,需要对这个目录进行一些调整:
- 将
vite.config.js
换成webpack.config.js
:因为我们要基于webpack5
对项目进行编译构建 - 初始化新的
package.json
:原本的package.json
依赖了一些我们不需要的内容,因此直接删除重新初始化 - 去除
index.html
中的<script type="module" src="/src/main.js"></script>
,因为这是vite
需要的,但webpack
并不需要
下面是调整后的目录结构:
安装基本依赖
下面列出了需要的一些基本依赖,其他的依赖在后面需要的时候在进行安装:
npm install vue@next
npm install webpack webpack-cli webpack-dev-server -D
对 webpack
进行基本配置
// webpack.config.jsconst path = require('path')module.exports = {entry: {path: './src/main.js'},output: {filename: 'assets/js/[name].[contenthash:6].js',path: path.resolve(__dirname, './dist')}
}
在 package.json
中创建脚本
// package.json"scripts": {"dev": "webpack server --mode=development","build": "webpack --mode=production"}
针对不同文件类型进行配置 webpack
启动项目
完后以上准备工作之后,我们就可以通过 npm run dev
命令来启动项目,但是你会发现这样的错误:
其实就是 webpack
不能认识 .vue
文件,它需要我们提供一个 loader
对其进行处理,这个 loader 就是官方文档中提到的:
其中 @vitejs/plugin-vue
这个是 vite
才需要的,因此我们只需要 vue-loader@next
、@vue/compiler-sfc
注意:vue-loader 默认是处理 vue2 的,这里使用的是 vue3,所以要安装 vue-loader@next
在 webpack
中处理 .vue
文件
首先通过 npm install vue-loader@next @vue/compiler-sfc -D
安装需要的依赖,然后在 webpack.config.js
中进行配置:
// webpack.config.jsconst path = require('path')
const { VueLoaderPlugin } = require('vue-loader')module.exports = {entry: {path: './src/main.js'},module: {rules: [{test: /\.vue$/,use: 'vue-loader'}]},output: {filename: 'assets/js/[name].[contenthash:6].js',path: path.resolve(__dirname, './dist')},plugins: [new VueLoaderPlugin(),]
}
配置完成之后,我们在通过 npm run dev
启动项目,不出意外的你将得到下面的错误:
显然 webpack
也不认识 <style></style>
中样式相关的内容,这一点大家都知道,那肯定是要使用 style-loader
和 css-loader
在 webpack
中处理 <style></style>
样式相关的内容
首先通过 npm install style-loader css-loader -D
安装需要的依赖,然后在 webpack.config.js
中进行配置:
// webpack.config.jsconst path = require('path')
const { VueLoaderPlugin } = require('vue-loader')module.exports = {entry: {path: './src/main.js',},module: {rules: [{test: /\.vue$/,use: 'vue-loader',},{test: /.css$/,use: ['style-loader', 'css-loader'],},],},output: {filename: 'assets/js/[name].[contenthash:6].js',path: path.resolve(__dirname, './dist'),},plugins: [new VueLoaderPlugin(),]
}
配置完成之后,又一次通过 npm run dev
启动项目,不出意外的你将又一次得到下面的错误:
意思就是说 webpack
仍然无法识别图片类型的文件,如:.(png|jpg|jpeg|gif)
等,那么到了这里相信你灵光一闪,想到了需要使用 file-loader/url-loader
,但在这我们不需要配置这两个 loader
,因为我们使用的是 webpack5
,是时候使用其中 资源模块(asset module) 的模块类型了.
配置 webpack
资源模块
废话不多说,直接上官方文档使用说明:
下面就是配置后的 webpack.config.js
内容:
const path = require('path')
const { VueLoaderPlugin } = require('vue-loader')module.exports = {entry: {path: './src/main.js',},module: {rules: [{test: /\.vue$/,use: 'vue-loader',},{test: /\.css$/,use: ['style-loader', 'css-loader'],},{test: /\.(png|jpe?g|gif)$/,type: 'asset/resource',generator: {filename: 'assets/img/[hash][ext]'}}],},output: {filename: 'assets/js/[name].[contenthash:6].js',path: path.resolve(__dirname, './dist'),},plugins: [new VueLoaderPlugin()]
}
这时候在通过 npm run dev
启动项目,发现终于没有报错了:
此时访问页面内容:
是不是会觉得
别慌问题不大,其实就是没有给 webpack
打包后的 js
代码指定模板.
为 webpack 指定模板
熟悉 webpack
的你肯定猜到要是用 html-webpack-plugin
插件,首先通过 npm install html-webpack-plugin -D
安装依赖,然后配置 webpack.config.js
文件:
const path = require('path')
const { VueLoaderPlugin } = require('vue-loader')
const HtmlWebpackPlugin = require('html-webpack-plugin')module.exports = {entry: {path: './src/main.js',},module: {rules: [{test: /\.vue$/,use: 'vue-loader',},{test: /\.css$/,use: ['style-loader', 'css-loader'],},{test: /\.(png|jpe?g|gif)$/,type: 'asset/resource',generator: {filename: 'assets/img/[hash][ext]'}}],},output: {filename: 'assets/js/[name].[contenthash:6].js',path: path.resolve(__dirname, './dist'),},plugins: [new VueLoaderPlugin(),new HtmlWebpackPlugin({template: path.resolve(__dirname, './index.html')})]
}
此时,在重新通过 npm run dev
启动项目并访问页面:
至此,终于成功配置了一个简单的基于 vue3
和 webpack5
的项目.
优化配置文件
通过 F12 打开页面控制台,你会看到这么一段警告信息:
其中 __VUE_OPTIONS_API__
和 __VUE_PROD_DEVTOOLS__
对应的值都是 Boolean
类型,分别代表的是:
__VUE_OPTIONS_API__
:表示是否支持options api
的写法,默认是true
__VUE_PROD_DEVTOOLS__
:表示生产包是否要继续支持devtools
插件,默认是false
即便它们都有默认值,可以不进行设置,但是 Vue
希望我们自己去设置这两个配置,毕竟如果完全拥抱 Vue3
的话,写法上没有必要在使用 options api
的格式,这样在打包的时候,包的体量上也会有所减少.
const path = require('path')
const { VueLoaderPlugin } = require('vue-loader')
const HtmlWebpackPlugin = require('html-webpack-plugin')
const { DefinePlugin } = require('webpack')module.exports = {entry: {path: './src/main.js',},module: {rules: [{test: /\.vue$/,use: 'vue-loader',},{test: /\.css$/,use: ['style-loader', 'css-loader'],},{test: /\.(png|jpe?g|gif)$/,type: 'asset/resource',generator: {filename: 'assets/img/[hash][ext]',},},],},output: {filename: 'assets/js/[name].[contenthash:6].js',path: path.resolve(__dirname, './dist'),},plugins: [new VueLoaderPlugin(),new HtmlWebpackPlugin({template: path.resolve(__dirname, './index.html'),}),new DefinePlugin({__VUE_PROD_DEVTOOLS__: false,__VUE_OPTIONS_API__: false,}),],
}
最后
首先通过 npm run build
查看打包后生成为目录结构:
目前存在的不足:
js
文件没有进行抽取,现在所有的js
内容都会默认打包到main.hash.js
中css
样式相关的内容没有进行抽取,现在的样式全部以<style></style>
标签的形式插入在html
文件中- 没有对
css
进行兼容处理 - 没有对
js
进行兼容处理 - 没有很好的区分各个环境,如:开发、测试、生产
- …
当然也有对应的处理方式:
- 通过
webpack
的optimization
选项配置,抽离对应的js
- 通过
mini-css-extract-plugin
插件来抽离对应的css
- 通过
postcss
对css
进行处理 - 通过
babel
对js
进行处理 - 针对公共配置部分进行抽取,或者通过环境变量去合成最终的配置项
- ···
以上的这些就不在一一进行配置,在需要时在进行相应配置即可.
至此,我们成功的将基于 vite + vue3
项目转换成了基于 webpack5 + vue3
的项目.
相关文章:

webpack5 + vue3 从零配置项目
前言 虽然在实际项目当中很少会从 0 到 1 配置一个项目,毕竟很多重复工作是没有必要的,脚手架将这些重复性的工作进行了整合,方便开发者使用。也正因如此,导致部分开发者过于依赖脚手架,却不清楚其内部的实现流程&…...

Queuing 表(buffer表)的优化实践 | OceanBase 性能优化实践
案例问题描述 该案例来自一个金融行业客户的问题:他们发现某个应用对一个数据量相对较小的表(仅包含数千条记录)访问时,频繁遇到性能下降的情况。为解决此问题,客户向我们求助进行分析。我们发现这张表有频繁的批量插…...
./mysqld: error while loading shared libraries: libaio.so.1: cannot open sha
mysql:5.6 使用离线方式安装:rpm -ivh --nodeps mysql* ,执行 systemctl start mysqld.service发现启动不了,通过vi /var/log/mysql.log看到如下关键字:libraries: libaio.so.1,之前也是按照网上帖子各种修改都没有解决…...

Qt主线程把数据发给子线程,主线程会阻塞吗
演示: #include <QCoreApplication> #include <QThread> #include <QObject> #include <QDebug>// 子线程类 class Worker : public QObject {Q_OBJECT public slots:void processData(int data) {qDebug() << "Processing dat…...

前后端、网关、协议方面补充
这里写目录标题 前后端接口文档简介前后端视角对于前端对于后端代码注册路由路由处理函数 关于httpGET/POST底层网络关于前端的获取 路由器网关路由器的IP简介公网IP(WAN IP)私网IP(LAN IP)无线网络IP(WIFI IP)查询路由器私网IP路由器公网IP LAN口与WIFI简介基本原理 手动配置电…...

如何在Mac上切换到JDK 17开发环境
在本文中,我将为您介绍如何在Mac上切换到JDK 17,包括下载和安装JDK 17、设置环境变量、在IntelliJ IDEA中配置项目、修改Maven编译配置,并最终使用mvn clean install重新编译项目。通过这个流程,您可以顺利地将开发环境升级到JDK …...
深入探索 TypeScript:从基础到高级特性
深入探索 TypeScript:从基础到高级特性 一、引言 在现代软件开发领域,TypeScript 已经成为了一种极具影响力的编程语言。它基于 JavaScript,并为其添加了强大的静态类型系统,使得代码在开发阶段就能进行更严格的类型检查&#x…...

Leetcode:118. 杨辉三角——Java数学法求解
题目——Leetcode:118. 杨辉三角 给定一个非负整数 numRows,生成「杨辉三角」的前 numRows 行。 在「杨辉三角」中,每个数是它左上方和右上方的数的和。 示例 1: 输入: numRows 5 输出: [[1],[1,1],[1,2,1],[1,3,3,1],[1,4,6,4,1]]示例 2: 输入: numRow…...

SHELL脚本(Linux)
声明 学习视频来自 B 站UP主泷羽sec,如涉及侵权马上删除文章。 笔记的只是方便各位师傅学习知识,以下网站只涉及学习内容,其他的都与本人无关,切莫逾越法律红线,否则后果自负。 ✍🏻作者简介:致…...

单元测试、集成测试、系统测试、验收测试、压力测试、性能测试、安全性测试、兼容性测试、回归测试(超详细的分类介绍及教学)
目录 1.单元测试 实现单元测试的方法: 注意事项: 2.集成测试 需注意事项: 实现集成测试的方法: 如何实现高效且可靠的集成测试: 3.系统测试 实现系统测试的方法: 须知注意事项: 4.验收测试 实现验…...
低代码集成多方API的简单实现
在现代软件开发中,集成多个API服务提供商已成为常见需求。然而,不同的API认证机制和数据格式使得集成过程变得复杂且耗时。为了应对这些挑战,本文将介绍一种低代码解决方案,通过配置化管理和简化的代码逻辑,帮助开发者…...

【测试框架篇】单元测试框架pytest(1):环境安装和配置
一、pytest简介 Pytest是Python的一种单元测试框架,与Python自带的unittest测试框架类似,但是比 unittest框架使用起来更简洁,效率更高。 二、pytest特点 Pytest是一个非常成熟的Python测试框架,主要特点有以下几点: 非常容易…...
Python数据分析NumPy和pandas(二十九、其他Python可视化工具)
与其他开源工具一样,在 Python 中创建图形有很多选项(太多了,无法一一列举)。自 2010 年以来,主要开发工作集中在创建用于在 Web 上发布交互式图形上。例如: Altair、Bokeh 和 Plotly 等工具,可…...

Unity中HDRP设置抗锯齿
一、以前抗锯齿的设置方式 【Edit】——>【Project Settings】——>【Quality】——>【Anti-aliasing】 二、HDRP项目中抗锯齿的设置方式 在Hierarchy中——>找到Camera对象——>在Inspector面板上——>【Camera组件】——>【Rendering】——>【Pos…...

Spring Boot实现文件上传与OSS集成:从基础到应用
目录 前言1. 文件上传的基础实现1.1 前端文件上传请求1.2 后端文件接收与保存 2. 集成第三方OSS服务2.1 准备工作2.2 编写OSS集成代码2.3 修改Controller实现文件上传至OSS 3. 文件上传的扩展:多文件上传与权限控制结语 前言 随着互联网应用的快速发展,…...

Python学习26天
集合 # 定义集合 num {1, 2, 3, 4, 5} print(f"num:{num}\nnum数据类型为:{type(num)}") # 求集合中元素个数 print(f"num中元素个数为:{len(num)}") # 增加集合中的元素 num.add(6) print(num) # {1,2,3,4,5,6} # 删除…...
linux startup.sh shutdown.sh (kkFileView)
linux启动脚本和关闭脚本startup.sh shutdown.sh (kkFileView) startup.sh DIR_HOME("/opt/openoffice.org3" "/opt/libreoffice" "/opt/libreoffice6.1" "/opt/libreoffice7.0" "/opt/libreoffice7.1&q…...
[MySQL]隐式类型转换
安全等号 <> 如果有参数为NULL,则除了相等比较运算符(),比较的结果为null。对于 nullnull,结果为true。 在select语句中,使用 时,结果不会包含值为 null 的记录,但如果使用安全等号 <> 来…...
面经总结1
文章目录 如何保证批量请求失败,只弹出一个toast1使用计数器:2使用标志变量: 如何减少项目里的if-else1使用多态2使用策略模式3使用字典映射4使用状态模式 babel-runtime 作用是啥如何实现 PDF 预览和下载1浏览器内置PDF阅读器2使用PDF.js库3…...

Oracle19C AWR报告分析之Instance Efficiency Percentages (Target 100%)
Oracle19C AWR报告分析之Instance Efficiency Percentages 一、分析数据二、详细分析2.1 Instance Efficiency Percentages (Target 100%)各项指标及其解释2.2 分析和总结 一、分析数据 二、详细分析 在 Oracle AWR (Automatic Workload Repository) 报告中,每个性能…...
浅谈 React Hooks
React Hooks 是 React 16.8 引入的一组 API,用于在函数组件中使用 state 和其他 React 特性(例如生命周期方法、context 等)。Hooks 通过简洁的函数接口,解决了状态与 UI 的高度解耦,通过函数式编程范式实现更灵活 Rea…...

stm32G473的flash模式是单bank还是双bank?
今天突然有人stm32G473的flash模式是单bank还是双bank?由于时间太久,我真忘记了。搜搜发现,还真有人和我一样。见下面的链接:https://shequ.stmicroelectronics.cn/forum.php?modviewthread&tid644563 根据STM32G4系列参考手…...
线程与协程
1. 线程与协程 1.1. “函数调用级别”的切换、上下文切换 1. 函数调用级别的切换 “函数调用级别的切换”是指:像函数调用/返回一样轻量地完成任务切换。 举例说明: 当你在程序中写一个函数调用: funcA() 然后 funcA 执行完后返回&…...

linux arm系统烧录
1、打开瑞芯微程序 2、按住linux arm 的 recover按键 插入电源 3、当瑞芯微检测到有设备 4、松开recover按键 5、选择升级固件 6、点击固件选择本地刷机的linux arm 镜像 7、点击升级 (忘了有没有这步了 估计有) 刷机程序 和 镜像 就不提供了。要刷的时…...

【SQL学习笔记1】增删改查+多表连接全解析(内附SQL免费在线练习工具)
可以使用Sqliteviz这个网站免费编写sql语句,它能够让用户直接在浏览器内练习SQL的语法,不需要安装任何软件。 链接如下: sqliteviz 注意: 在转写SQL语法时,关键字之间有一个特定的顺序,这个顺序会影响到…...

华为OD机试-食堂供餐-二分法
import java.util.Arrays; import java.util.Scanner;public class DemoTest3 {public static void main(String[] args) {Scanner in new Scanner(System.in);// 注意 hasNext 和 hasNextLine 的区别while (in.hasNextLine()) { // 注意 while 处理多个 caseint a in.nextIn…...
Java 加密常用的各种算法及其选择
在数字化时代,数据安全至关重要,Java 作为广泛应用的编程语言,提供了丰富的加密算法来保障数据的保密性、完整性和真实性。了解这些常用加密算法及其适用场景,有助于开发者在不同的业务需求中做出正确的选择。 一、对称加密算法…...

HBuilderX安装(uni-app和小程序开发)
下载HBuilderX 访问官方网站:https://www.dcloud.io/hbuilderx.html 根据您的操作系统选择合适版本: Windows版(推荐下载标准版) Windows系统安装步骤 运行安装程序: 双击下载的.exe安装文件 如果出现安全提示&…...
C#学习第29天:表达式树(Expression Trees)
目录 什么是表达式树? 核心概念 1.表达式树的构建 2. 表达式树与Lambda表达式 3.解析和访问表达式树 4.动态条件查询 表达式树的优势 1.动态构建查询 2.LINQ 提供程序支持: 3.性能优化 4.元数据处理 5.代码转换和重写 适用场景 代码复杂性…...

c++第七天 继承与派生2
这一篇文章主要内容是 派生类构造函数与析构函数 在派生类中重写基类成员 以及多继承 第一部分:派生类构造函数与析构函数 当创建一个派生类对象时,基类成员是如何初始化的? 1.当派生类对象创建的时候,基类成员的初始化顺序 …...