前端工程化指的是:在企业级的前端项目开发中,把前端开发所需要的工具、技术、流程、经验等进行规范化、标准化。
目前主流的前端工程化解决方案:
概念:webpack 是前端工程化的具体解决方案
主要功能:提供了友好的前端模块化开发支持,以及代码压缩混淆、处理浏览器端 JavaScript 的兼容性、性能优化等强大功能。
ES6 模块:
https://www.runoob.com/w3cnote/es6-module.html
jquery语法:
https://www.runoob.com/jquery/jquery-syntax.html
Html:
DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Documenttitle>
head>
<body>
<ul>
<li>这是第 1 个lili>
<li>这是第 2 个lili>
<li>这是第 3 个lili>
<li>这是第 4 个lili>
<li>这是第 5 个lili>
<li>这是第 6 个lili>
<li>这是第 7 个lili>
<li>这是第 8 个lili>
<li>这是第 9 个lili>
ul>
<script src="index.js">script>
body>
html>
js:
// 1 使用 ES6 导入语法,导入 jQuery
import $ from 'jquery'
// 2. 定义 jQuery 入口函数
$(function () {
// 3. 实现奇偶行变色
$('li:odd').css();
$('li:even').css("background-color", "pink");
})
运行报错效果无法实现:Uncaught SyntaxError: Cannot use import statement outside a module
原因:浏览器不支持 **import $ from ‘jquery’**代码
解决方法:用webpack进行兼容性适配
在终端运行如下命令,安装 webpack 相关的两个包:
npm install [email protected] [email protected] -D
-D 是 --save-dev 的简写表示将 包名和版本记录到 devDependencies 节点下
package.json
{
"name": "webpack1",
"version": "1.0.0",
"description": "",
"main": "index.js",
"scripts": {
"test": "echo \"Error: no test specified\" && exit 1"
},
"keywords": [],
"author": "",
"license": "ISC",
"dependencies": {
"jquery": "^3.6.1"
},
"devDependencies": {
"webpack": "^5.42.1",
"webpack-cli": "^4.7.2"
}
}
dependencies节点表示开发和上线时都要用到的包,devDependencies表示开发时需要用到的包,项目上线时不需要用到。
https://npmjs.com/package
https://www.liaoxuefeng.com/wiki/1022910821149312/1023027697415616
// 使用 Node.js 中的导出语法,导出一个 webpack 的配置对象
module.exports = {
mode: "development" // mode 来指定构建模式,可选值有 development 和 production
}
"scripts": {
"dev": "webpack"
},
DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta http-equiv="X-UA-Compatible" content="IE=edge">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Documenttitle>
head>
<body>
<ul>
<li>这是第 1 个lili>
<li>这是第 2 个lili>
<li>这是第 3 个lili>
<li>这是第 4 个lili>
<li>这是第 5 个lili>
<li>这是第 6 个lili>
<li>这是第 7 个lili>
<li>这是第 8 个lili>
<li>这是第 9 个lili>
ul>
<script src="../dist/main.js">script>
body>
html>
mode 的可选值有两个:
webpack.config.js 是 webpack 的配置文件。webpack 在真正开始打包构建之前,会先读取整个配置文件,从而基于给定的配置,对项目进行打包。
注意:由于 webpack 是基于 node.js 开发出来的打包工具,因此在它的配置文件中,支持使用 node.js 相关的语法和模块进行 webpack 的个性化配置。
注意:可以在 webpack.config.js 中修改打包的默认约定
在 webpack.config.js 配置文件中,通过 entry 节点指定 打包的入口。通过 output 节点指定打包的出口
示例代码:
const path = require('path')
// 使用 Node.js 中的导出语法,导出一个 webpack 的配置对象
module.exports = {
mode: "development", // mode 来指定构建模式,可选值有 development 和 production
// entry 指定要处理哪个文件
entry: path.join(__dirname, './src/index1.js'),
// 指定生成的文件要存放到到哪里
output: {
// 存放目录
path: path.join(__dirname, "./dist"),
// 生成的文件名
filename: 'bundle.js'
}
}
通过安装第三方插件,可以扩展 webpack 的能力,从而让 webpack 用起来更方便。最常用的 webpack 的 webpack 插件有如下两个:
运行如下命令,即可在项目中安装此插件
npm install [email protected] -D
"scripts": {
"dev": "webpack serve"
}
在实际开发中,webpack 默认只能通过打包处理以 .js 后缀名结尾的模块,其它的模块需要调用 loader 加载器才能正常打包
loader 加载器的作用:
npm i [email protected] [email protected] -D
module: {
rules: [
// 定义了不同模块对应的 loader
{test: /\.css$/, use: ['style-loader', 'css-loader']}
]
}
其中,test表示匹配的文件类型,ues 表示要调用的loader
注意:
npm i [email protected] [email protected] -D
module: {
rules: [
{test: /\.less$/, ues: ['style-loader', 'css-loader', 'less-loader']}
]
}
用base64图片的优点:可以防止发送一些不必要的网络请求
缺点:图片体积会变大
步骤:
npm i url-loader@4.1.1 file-loader@6.2.0 -D
module: {
rules: [
// 定义了不同模块对应的 loader
{test: /\.jpg|png|gif$/, use: "url-loader?limit=22229"}
]
}
其中 ? 之后的是 loader 的参数项:
webpack 只能处理一部分高级的 JavaScript 语法。对于那些 webpack 不能处理的高级 js语法,需要借助于 babel-loader 进行打包处理。例如 webpack 无法处理下面的 JavaScript 代码:
// 1.定义了名为 info 的装饰器
function info(target) {
// 2.为目标添加静态属性 info
target.info = "Person info";
}
// 3.为 Person 类应用 info 装饰器
@info
class Person {}
// 4.打印 Perosn 的静态属性 info
console.log(Person.info);
运行如下的命令安装对应的依赖包:
npm i [email protected] @babel/[email protected] @babel/[email protected] -D
在 webpack.config.js 的 module -> rules 数组中,添加 loader 规则如下:
// 注意:必须使用 exclude 指定排除项:因为 node_modules 目录下的第三方包不需要被打包
{test: /\.js$/, use: "babel-loader", exclude: /node_modules/}
在项目根目录下,创建名为 babel.config.js 的配置文件,定义 babel 的配置项 如下:
module.exports = {
// 声明 babel 可用的插件
// 将来,webpack 在调用 babel-loader 的时候,会先加载 plugins 插件来使用
plugins: [["@babel/plugin-proposal-decorators", { legacy: true }]]
}
在 package.json 文件的 script 节点下,新增 build 命令如下:
"scripts": {
"dev": "webpack",
"build": "webpack --mode production"
},
–model 是一个参数项,用来指定 webpack 的运行模式,production 代表生产环境,会对打包生成的文件进行代码压缩和性能优化。
注意:通过 --model 指定的参数项目,会覆盖 **weback.config.js **中的 model 选项。
在 **webpack.config.js **配置文件中的 **output **节点中,进行如下配置:
output: {
// 存放目录
path: path.join(__dirname, "dist"),
// 生成的文件名
filename: 'js/bundle.js'
}
在 **webpack.config.js **中的 url-loader 配置项目,新增 outputPath 选项即可指定图片文件的输出路径:
// 处理图片文件的loader
// 在配置 url-laoder 的时候,多个参数之间使用 & 符号进行分隔
{test: /\.jpg|png|gif$/, use: "url-loader?limit=22229&outputPath=images"},
为了在每次打包发布时自动清理掉 dist 目录中的旧文件,可以安装并配置 claen-webpack-plugin 插件:
https://www.npmjs.com/package/clean-webpack-plugin
安装:
npm install --save-dev clean-webpack-plugin
配置 webpack.config.js
// 注意:左侧的{ }是解构赋值
const { CleanWebpackPlugin } = require('clean-webpack-plugin');
// 插件的数组,将来 webpack 在运行时,会加载并调用这些插件
plugins: [new CleanWebpackPlugin()]
Source Map 就是一个信息文件,里面存储着位置信息。也就是说,Source Map文件中存储着压缩混淆后的代码,所对应的转换前的位置。
有了它,出错的时候,除错工具将直接显示原始代码,而不是转换后的代码,能够极大的方便后期的调试。
开发环境下默认生成的 Source Map,记录的是生成后的代码的位置。会导致运行时报错的行数与源代码的行数不一致的问题。
开发环境下,推荐在 webpack.config.js 中添加下配置,即可保证运行时报错的行数与源码的行数保持一致:
const path = require('path')
// 注意:左侧的{ }是解构赋值
const { CleanWebpackPlugin } = require('clean-webpack-plugin');
// 使用 Node.js 中的导出语法,导出一个 webpack 的配置对象
module.exports = {
mode: "development", // mode 来指定构建模式,可选值有 development 和 production
// entry 指定要处理哪个文件
entry: path.join(__dirname, './src/index1.js'),
// 指定生成的文件要存放到到哪里
output: {
// 存放目录
path: path.join(__dirname, "dist"),
// 生成的文件名
filename: 'js/bundle.js'
},
devServer: {
contentBase: "./",
port: 8081
},
module: {
rules: [
// 定义了不同模块对应的 loader
{test: /\.css$/, use: ['style-loader', 'css-loader']},
// 处理 .less 文件的loader
{test: /\.less$/, use: ['style-loader', 'css-loader', 'less-loader']},
// 处理图片文件的loader
// 在配置 url-laoder 的时候,多个参数之间使用 & 符号进行分隔
{test: /\.jpg|png|gif$/, use: "url-loader?limit=22229&outputPath=images"},
// 注意:必须使用 exclude 指定排除项:因为 node_modules 目录下的第三方包不需要被打包
{test: /\.js$/, use: "babel-loader", exclude: /node_modules/}
]
},
// eval-source-map 仅限在“开发模式”下使用,不建议在“生产模式”下使用
// 此选项生成的 Source Map 能够保证“运行时报错的行数”与“源码的行数”保持一致
devtool: "eval-source-map",
// 插件的数组,将来 webpack 在运行时,会加载并调用这些插件
plugins: [new CleanWebpackPlugin()]
}
在生产环境下,如果省略了devtool 选项,最终生成的文件中不包含 Source Map。这额能够防止源代码通过Source Map 的形式暴露给别有所图的人。
在生产环境下,如果只想定位报错的具体行数,且不想暴露源码。此时可以将 devtool 的值设置为 nosources-source-map。
const path = require('path')
// 注意:左侧的{ }是解构赋值
const { CleanWebpackPlugin } = require('clean-webpack-plugin');
// 使用 Node.js 中的导出语法,导出一个 webpack 的配置对象
module.exports = {
mode: "development", // mode 来指定构建模式,可选值有 development 和 production
// entry 指定要处理哪个文件
entry: path.join(__dirname, './src/index1.js'),
// 指定生成的文件要存放到到哪里
output: {
// 存放目录
path: path.join(__dirname, "dist"),
// 生成的文件名
filename: 'js/bundle.js'
},
devServer: {
contentBase: "./",
port: 8081
},
module: {
rules: [
// 定义了不同模块对应的 loader
{test: /\.css$/, use: ['style-loader', 'css-loader']},
// 处理 .less 文件的loader
{test: /\.less$/, use: ['style-loader', 'css-loader', 'less-loader']},
// 处理图片文件的loader
// 在配置 url-laoder 的时候,多个参数之间使用 & 符号进行分隔
{test: /\.jpg|png|gif$/, use: "url-loader?limit=22229&outputPath=images"},
// 注意:必须使用 exclude 指定排除项:因为 node_modules 目录下的第三方包不需要被打包
{test: /\.js$/, use: "babel-loader", exclude: /node_modules/}
]
},
// eval-source-map 仅限在“开发模式”下使用,不建议在“生产模式”下使用
// 此选项生成的 Source Map 能够保证“运行时报错的行数”与“源码的行数”保持一致
// 在实际发布的时候,建议把 devtool 的值设置为 nosources-source-map 或者直接关闭 SourceMap
devtool: "nosources-source-map",
// 插件的数组,将来 webpack 在运行时,会加载并调用这些插件
plugins: [new CleanWebpackPlugin()]
}
@ 符号表示 scr 源代码目录,从外往里查找;不要使用 …/ 从里往外找
在配置文件中设置(添加resolve节点):
const path = require('path')
// 注意:左侧的{ }是解构赋值
const { CleanWebpackPlugin } = require('clean-webpack-plugin');
// 使用 Node.js 中的导出语法,导出一个 webpack 的配置对象
module.exports = {
mode: "development", // mode 来指定构建模式,可选值有 development 和 production
// entry 指定要处理哪个文件
entry: path.join(__dirname, './src/index1.js'),
// 指定生成的文件要存放到到哪里
output: {
// 存放目录
path: path.join(__dirname, "dist"),
// 生成的文件名
filename: 'js/bundle.js'
},
devServer: {
contentBase: "./",
port: 8081
},
module: {
rules: [
// 定义了不同模块对应的 loader
{test: /\.css$/, use: ['style-loader', 'css-loader']},
// 处理 .less 文件的loader
{test: /\.less$/, use: ['style-loader', 'css-loader', 'less-loader']},
// 处理图片文件的loader
// 在配置 url-laoder 的时候,多个参数之间使用 & 符号进行分隔
{test: /\.jpg|png|gif$/, use: "url-loader?limit=22229&outputPath=images"},
// 注意:必须使用 exclude 指定排除项:因为 node_modules 目录下的第三方包不需要被打包
{test: /\.js$/, use: "babel-loader", exclude: /node_modules/}
]
},
// eval-source-map 仅限在“开发模式”下使用,不建议在“生产模式”下使用
// 此选项生成的 Source Map 能够保证“运行时报错的行数”与“源码的行数”保持一致
// 在实际发布的时候,建议把 devtool 的值设置为 nosources-source-map 或者直接关闭 SourceMap
devtool: "nosources-source-map",
resolve: {
alias: {
// 告诉 webpack, 程序员写的代码中,@符号表示 src 这一层目录
'@': path.join(__dirname, "/src/")
}
},
// 插件的数组,将来 webpack 在运行时,会加载并调用这些插件
plugins: [new CleanWebpackPlugin()]
}
https://chrome.google.com/webstore/detail/vuejs-devtools/nhdogjmejiglipccpnnnanhbledajbpd/related?hl=zh
安装插件,打开允许访问文件网址