webpack onePage
如何在浏览器端实现模块
浏览器端的模块化
问题:
- 效率问题:精细的模块划分带来了更多的JS文件,更多的JS文件带来了更多的请求,降低了页面访问效率
- 兼容性问题:浏览器目前仅支持ES6的模块化标准,并且还存在兼容性问题
- 工具问题:浏览器不支持npm下载的第三方包
这些仅仅是前端工程化的一个缩影
当开发一个具有规模的程序,你将遇到非常多的非业务问题,这些问题包括:执行效率、兼容性、代码的可维护性可扩展性、团队协作、测试等等等等,我们将这些问题称之为工程问题。工程问题与业务无关,但它深刻的影响到开发进度,如果没有一个好的工具解决这些问题,将使得开发进度变得极其缓慢,同时也让开发者陷入技术的泥潭。
根本原因
思考:上面提到的问题,为什么在node端没有那么明显,反而到了浏览器端变得如此严重呢?
答:,它的效率比浏览器远程传输文件高的多
根本原因:在浏览器端,开发时态(devtime)和运行时态(runtime)的侧重点不一样
开发时态,devtime:
- 模块划分越细越好
- 支持多种模块化标准
- 支持npm或其他包管理器下载的模块
- 能够解决其他工程化的问题
运行时态,runtime:
- 文件越少越好
- 文件体积越小越好
- 代码内容越乱越好
- 所有浏览器都要兼容
- 能够解决其他运行时的问题,主要是执行效率问题
这种差异在小项目中表现的并不明显,可是一旦项目形成规模,就越来越明显,如果不解决这些问题,前端项目形成规模只能是空谈
解决办法
既然开发时态和运行时态面临的局面有巨大的差异,因此,我们需要有一个工具,这个工具能够让开发者专心的在开发时态写代码,然后利用这个工具将开发时态编写的代码转换为运行时态需要的东西。
这样的工具,叫做构建工具

这样一来,开发者就可以专注于开发时态的代码结构,而不用担心运行时态遇到的问题了。
常见的构建工具
- webpack
- grunt
- gulp
- browserify
- fis
- 其他
webpack的安装和使用
webpack官网:https://www.webpackjs.com/
webpack简介
全部东西视为模块
webpack是基于模块化的打包(构建)工具,它把一切视为模块
它通过一个开发时态的入口模块为起点,分析出所有的依赖关系,然后经过一系列的过程(压缩、合并),最终生成运行时态的文件。
webpack的特点:
- 为前端工程化而生:webpack致力于解决前端工程化,特别是浏览器端工程化中遇到的问题,让开发者集中注意力编写业务代码,而把工程化过程中的问题全部交给webpack来处理
- 简单易用:支持零配置,可以不用写任何一行额外的代码就使用webpack
- 强大的生态:webpack是非常灵活、可以扩展的,webpack本身的功能并不多,但它提供了一些可以扩展其功能的机制,使得一些第三方库可以融于到webpack中
- 基于nodejs:由于webpack在构建的过程中需要读取文件,因此它是(
理解:中间打包需要node环境,如果左边是浏览器环境,右边即浏览器;如果左边是node环境,右边即node
- 基于模块化:webpack在构建过程中要分析依赖关系,方式是通过模块化导入语句进行分析的,它支持各种模块化标准,包括CommonJS、ES6 Module
只打包依赖的东西
webpack的安装
webpack通过npm安装,它提供了两个包:
- webpack:核心包,包含了webpack构建过程中要用到的所有api
- webpack-cli:提供一个简单的cli命令,它调用了webpack核心包的api,来完成构建过程
安装方式:
- 全局安装:可以全局使用webpack命令,但是无法为不同项目对应不同的webpack版本
- 本地安装:推荐,每个项目都使用自己的webpack版本进行构建
先初始化npm init,在npm i -D webpack webpack-cli D开发依赖,而不是生产,右边要运行,右边已经构建完成,和webpack没关系了
使用
webpack
默认情况下,webpack会以./src/index.js作为入口文件分析依赖关系,打包到./dist/main.js文件中
通过--mode选项可以控制webpack的打包结果的运行环境
npx webpack命令运行
打包完成的代码要在什么环境内运行?
开发环境配置:npx webpack --mode=development
生产环境配置:npx webpack --mode=production
- 也可以在package.json配置
"scripts": {
"build": "webpack --mode=production",
"dev": "webpack --mode=development"
},
运行:npm run build--------生产环境
npm run dev----------开发环境- 也可以在webpack.config.js里面配置以下代码:
module.exports = {
mode: 'development',
};模块化兼容性
由于webpack同时支持CommonJS和ES6 module,因此需要理解它们互操作时webpack是如何处理的
同模块化标准
如果导出和导入使用的是同一种模块化标准,打包后的效果和之前学习的模块化没有任何差异


不同模块化标准
不同的模块化标准,webpack按照如下的方式处理


命令实现:es6导出,commonjs导入
package.json配置好
"scripts": {
"build": "webpack --mode=production",
"dev": "webpack --mode=development"
},写好代码
index.js
var obj = require("./es6a")
console.log(obj);
console.log(obj.a, obj.b, obj.default)es6a.js
export var a = 1;
export var b = 2;
export default 3;es6导出,commonjs导入
common:
module.exports = {
a: 1,
b: 2,
c: 3
}index
import * as obj from "./commonjsa"
// import obj from "./commonjsa"
console.log(obj)最佳实践
代码编写最忌讳的是精神分裂,选择一个合适的模块化标准,然后贯彻整个开发阶段。
数字特效
配置文件
命令npx webpack打包
通过命令给与参数:npx webpack --mode=development也能完成打包、
webpack提供的cli支持很多的参数,例如--mode,但更多的时候,我们会使用更加灵活的配置文件来控制webpack的行为
默认情况下,webpack会读取webpack.config.js文件作为配置文件,但也可以通过CLI参数--config来指定某个配置文件
命令:npx webpack --config 123.js
配置文件中一个对象对象中的各种属性对应不同的webpack配置
module.exports = {}当命令行参数与配置文件中的配置出现冲突时,以命令行参数为准。
基本配置:
- mode:编译模式,字符串,取值为development或production,指定编译结果代码运行的环境,会影响webpack对编译结果代码格式的处理
在webpack.config.js里
module.exports = {
mode: "production"
}- entry:入口,字符串(后续会详细讲解),指定入口文件
默认./src/index.js
module.exports = {
entry:"./123.js"
}- output:出口,对象(后续会详细讲解),指定编译结果文件
默认./dist/main.js
module.exports = {
output:{
filename:"bundle.js"
}
}devtool配置
source map 源码地图
本小节的知识与 webpack 无关
前端发展到现阶段,很多时候都不会直接运行源代码,可能需要对源代码进行合并、压缩、转换等操作,真正运行的是转换后的代码

这就给调试带来了困难,因为当运行发生错误的时候,我们更加希望能看到源代码中的错误,而不是转换后代码的错误
jquery压缩后的代码:https://code.jquery.com/jquery-3.4.1.min.js
为了解决这一问题,chrome浏览器率先支持了source map,其他浏览器纷纷效仿,目前,几乎所有新版浏览器都支持了source map
source map实际上是一个配置,配置中不仅记录了所有源码内容,还记录了和转换后的代码的对应关系
下面是浏览器处理source map的原理


最佳实践:
- source map 应在开发环境中使用,作为一种调试手段
- source map ,source map的文件一般较大,不仅会导致额外的网络传输,还容易暴露原始代码。即便要在生产环境中使用source map,用于调试真实的代码运行问题,也要做出一些处理规避网络传输和代码暴露的问题。
webpack中的source map
使用 webpack 编译后的代码难以调试,可以通过 devtool 配置来优化调试体验
具体的配置见文档:https://www.webpackjs.com/configuration/devtool/
使用开发环境:
module.exports = {
mode: "development",
}
浏览器自带
eval("var obj = null;\nobj.abc();\nconsole.log(\"a module\")\n\n//# sourceURL=webpack:///./src/a.js?");eval是简易版的source map
生产环境production就不行,要想有则需要 devtool 配置
module.exports = {
mode: "production",
devtool: "eval"
}
webpack 的作用是将源代码编译(构建、打包)成最终代码

整个过程大致分为三个步骤
- 初始化
- 编译
- 输出

初始化
此阶段,webpack会将CLI参数、配置文件、默认配置进行融合,形成一个最终的配置对象。
对配置的处理过程是依托一个第三方库yargs完成的
此阶段相对比较简单,主要是为接下来的编译阶段做必要的准备
目前,可以简单的理解为,初始化阶段主要用于产生一个最终的配置
编译
- 创建chunk
chunk是webpack在内部构建过程中的一个概念,译为块,它表示通过。
根据入口模块(默认为./src/index.js)创建一个chunk

每个chunk都有至少两个属性:
- name:默认为main
- id:唯一编号,开发环境和name相同,生产环境是一个数字,从0开始
- 构建所有依赖模块



形成表格

递归加载./src/a.js

记录下来,保存到模块列表中

递归加载./src/b.js。这里加载的b是由于a依赖b,实际上b还没有加载

记录下来,保存到模块列表中

a加载完毕,继续加载index,到了b

AST在线测试工具:https://astexplorer.net/
简图

- 产生chunk assets
在第二步完成后,chunk中会产生一个模块列表,列表中包含了模块id和模块转换后的代码
接下来,webpack会根据配置为chunk生成一个资源列表,即chunk assets,资源列表可以理解为是生成到最终文件的文件名和文件内容

chunk hash是根据所有chunk assets的内容生成的一个hash字符串
hash:一种算法,具体有很多分类,特点是将一个任意长度的字符串转换为一个固定长度的字符串,而且可以保证原始内容不变,产生的hash字符串就不变
简图

- 合并chunk assets
将多个chunk的assets合并到一起,并产生一个总的hash

输出
此步骤非常简单,webpack将利用node中的fs模块(文件处理模块),根据编译产生的总的assets,生成相应的文件。

总过程


涉及术语
- module:模块,分割的代码单元,webpack中的模块可以是任何内容的文件,不仅限于JS
- chunk:webpack内部构建模块的块,一个chunk中包含多个模块,这些模块是从入口模块通过依赖分析得来的
- bundle:chunk构建好模块后会生成chunk的资源清单,清单中的每一项就是一个bundle,可以认为bundle就是最终生成的文件
- hash:最终的资源清单所有内容联合生成的hash值
- chunkhash:chunk生成的资源清单内容联合生成的hash值
- chunkname:chunk的名称,如果没有配置则使用main
- id:通常指chunk的唯一编号,如果在开发环境下构建,和chunkname相同;如果是生产环境下构建,则使用一个从0开始的数字进行编号
入口和出口

node内置模块 - path: https://nodejs.org/dist/latest-v12.x/docs/api/path.html
出口
这里的出口是针对资源列表的文件名或路径的配置
出口通过output进行配置
入口
入口真正配置的是chunk
入口通过entry进行配置
规则:
- name:chunkname
- hash: 总的资源hash,
- chunkhash: 使用
- id: 使用chunkid,不推荐。会导致生产环境和开发环境的名字不一致。
入口和出口的最佳实践
具体情况具体分析
下面是一些经典场景
一个页面一个JS

源码结构
|—— src
|—— pageA 页面A的代码目录
|—— index.js 页面A的启动模块
|—— ...
|—— pageB 页面B的代码目录
|—— index.js 页面B的启动模块
|—— ...
|—— pageC 页面C的代码目录
|—— main1.js 页面C的启动模块1 例如:主功能
|—— main2.js 页面C的启动模块2 例如:实现访问统计的额外功能
|—— ...
|—— common 公共代码目录
|—— ...webpack配置
module.exports = {
entry:{
pageA: "./src/pageA/index.js",
pageB: "./src/pageB/index.js",
pageC: ["./src/pageC/main1.js", "./src/pageC/main2.js"]
},
output:{
filename:"[name].[chunkhash:5].js"
}
}这种方式适用于页面之间的功能差异巨大、公共代码较少的情况,这种情况下打包出来的最终代码不会有太多重复
面试题:打包出来的js,里面会有公共代码,这里代码的重复会造成什么影响?
导致传输量增加。


一个页面多个JS

源码结构
|—— src
|—— pageA 页面A的代码目录
|—— index.js 页面A的启动模块
|—— ...
|—— pageB 页面B的代码目录
|—— index.js 页面B的启动模块
|—— ...
|—— statistics 用于统计访问人数功能目录
|—— index.js 启动模块
|—— ...
|—— common 公共代码目录
|—— ...webpack配置
module.exports = {
entry:{
pageA: "./src/pageA/index.js",
pageB: "./src/pageB/index.js",
statistics: "./src/statistics/index.js"//statistics和AB都没关系,可以单独开一个chunk
},
output:{
filename:"[name].[chunkhash:5].js"
}
}这种方式适用于页面之间有一些独立、相同的功能,专门使用一个chunk抽离这部分JS有利于浏览器更好的缓存这部分内容。
思考:为什么不使用多启动模块的方式?
单页应用
所谓单页应用,是指整个网站(或网站的某一个功能块)只有一个页面,页面中的内容全部靠JS创建和控制。 vue和react都是实现单页应用的利器。

源码结构
|—— src
|—— subFunc 子功能目录
|—— ...
|—— subFunc 子功能目录
|—— ...
|—— common 公共代码目录
|—— ...
|—— index.jswebpack配置
module.exports = {
entry: "./src/index.js",
output:{
filename:"index.[hash:5].js"
}
}loader
webpack做的事情,仅仅是分析出各种模块的依赖关系,然后形成资源列表,最终打包生成到指定的文件中。
更多的功能需要借助webpack loaders和webpack plugins完成。
webpack loader: loader本质上是一个函数,它的作用是将某个源码字符串转换成另一个源码字符串返回。

loader函数的将在模块解析的过程中被调用,以得到最终的源码。
全流程:

chunk中解析模块的流程:

chunk中解析模块的更详细流程:

处理loaders流程:

loader配置:
完整配置
module.exports = {
module: { //针对模块的配置,目前版本只有两个配置,rules、noParse
rules: [ //模块匹配规则,可以存在多个规则
{ //每个规则是一个对象
test: /\.js$/, //匹配的模块正则
use: [ //匹配到后应用的规则模块
{ //其中一个规则
loader: "模块路径", //loader模块的路径,该字符串会被放置到require中
options: { //向对应loader传递的额外参数
}
}
]
}
]
}
}简化配置
module.exports = {
module: { //针对模块的配置,目前版本只有两个配置,rules、noParse
rules: [ //模块匹配规则,可以存在多个规则
{ //每个规则是一个对象
test: /\.js$/, //匹配的模块正则
use: ["模块路径1", "模块路径2"]//loader模块的路径,该字符串会被放置到require中
}
]
}
}plugin
loader的功能定位是转换代码,而一些其他的操作难以使用loader完成,比如:
- 当webpack生成文件时,顺便多生成一个说明描述文件
- 当webpack编译启动时,控制台输出一句话表示webpack启动了
- 当xxxx时,xxxx
这种类似的功能需要把功能嵌入到webpack的编译流程中,而这种事情的实现是依托于plugin的

plugin的本质是一个带有apply方法的对象
var plugin = {
apply: function(compiler){
}
}通常,习惯上,我们会将该对象写成构造函数的模式
class MyPlugin{
apply(compiler){
}
}
var plugin = new MyPlugin();要将插件应用到webpack,需要把插件对象配置到webpack的plugins数组中,如下:
module.exports = {
plugins:[
new MyPlugin()
]
}compiler对象是在初始化阶段构建的,整个webpack打包期间只有一个compiler对象,后续完成打包工作的是compiler对象内部创建的compilation
apply方法会在创建好compiler对象后调用,并向方法传入一个compiler对象

compiler只有一个,compilation可能多个
compiler对象提供了大量的钩子函数(hooks,可以理解为事件),plugin的开发者可以注册这些钩子函数,参与webpack编译和生成。
你可以在apply方法中使用下面的代码注册钩子函数:
class MyPlugin{
apply(compiler){
compiler.hooks.事件名称.事件类型(name, function(compilation){
//事件处理函数
})
}
}事件名称
即要监听的事件名,即钩子名,所有的钩子:https://www.webpackjs.com/api/compiler-hooks
事件类型
这一部分使用的是 Tapable API,这个小型的库是一个专门用于钩子函数监听的库。
它提供了一些事件类型:
- tap:注册一个同步的钩子函数,函数运行完毕则表示事件处理结束
- tapAsync:注册一个基于回调的异步的钩子函数,函数通过调用一个回调表示事件处理结束
- tapPromise:注册一个基于Promise的异步的钩子函数,函数通过返回的Promise进入已决状态表示事件处理结束
处理函数
处理函数有一个事件参数compilation
区分环境
有些时候,我们需要针对生产环境和开发环境分别书写webpack配置
以前可以用这种方式来区分:

为了更好的适应这种要求,webpack允许配置不仅可以是一个对象,还可以是一个函数
module.exports = env => {
return {
//配置内容
}
}在开始构建时,webpack如果发现配置是一个函数,会调用该函数,将函数返回的对象作为配置内容,因此,开发者可以根据不同的环境返回不同的对象
在调用webpack函数时,webpack会向函数传入一个参数env,该参数的值来自于webpack命令中给env指定的值,例如
npx webpack --env abc # env: "abc"
npx webpack --env.abc # env: {abc:true}
npx webpack --env.abc=1 # env: {abc:1}
npx webpack --env.abc=1 --env.bcd=2 # env: {abc:1, bcd:2}这样一来,我们就可以在命令中指定环境,在代码中进行判断,根据环境返回不同的配置结果。
其他细节配置
context
context: path.resolve(__dirname, "app")该配置会影响入口和loaders的解析,入口和loaders的相对路径会以context的配置作为基准路径,这样,你的配置会独立于CWD(current working directory 当前执行路径)
output
library
library: "abc"这样一来,打包后的结果中,会将自执行函数的执行结果暴露给abc
libraryTarget
libraryTarget: "var"该配置可以更加精细的控制如何暴露入口包的导出结果
其他可用的值有:
- var:默认值,暴露给一个普通变量
- window:暴露给window对象的一个属性
- this:暴露给this的一个属性
- global:暴露给global的一个属性
- commonjs:暴露给exports的一个属性
- 其他:https://www.webpackjs.com/configuration/output/#output-librarytarget
target
target:"web" //默认值设置打包结果最终要运行的环境,常用值有
- web: 打包后的代码运行在web环境中
- node:打包后的代码运行在node环境中
- 其他:https://www.webpackjs.com/configuration/target/
module.noParse
noParse: /jquery/不解析正则表达式匹配的模块,通常用它来忽略那些,以提高构建性能。和运行性能无关。
resolve
resolve的相关配置主要用于控制模块解析过程
modules
modules: ["node_modules"] //默认值。可以自行配置更改当解析模块时,如果遇到导入语句,require("test"),webpack会从下面的位置寻找依赖的模块
- 当前目录下的
node_modules目录 - 上级目录下的
node_modules目录 - ...
extensions
extensions: [".js", ".json"] //默认值- test.js
- test.json
alias
alias: {
"@": path.resolve(__dirname, 'src'),
"_": __dirname
}有了alias(别名)后,导入语句中可以加入配置的键名,例如require("@/abc.js"),webpack会将其看作是require(src的绝对路径+"/abc.js")。
在大型系统中,源码结构往往比较深和复杂,别名配置可以让我们更加方便的导入依赖
externals
externals: { jquery: "$", lodash: "_"}从最终的bundle中排除掉配置的配置的源码,例如,入口模块是
//index.jsrequire("jquery")require("lodash")生成的bundle是:
(function(){
...
})({
"./src/index.js": function(module, exports, __webpack_require__){
__webpack_require__("jquery")
__webpack_require__("lodash")
},
"jquery": function(module, exports){
//jquery的大量源码
},
"lodash": function(module, exports){
//lodash的大量源码
},
})但有了上面的配置后,则变成了
(function(){
...
})({
"./src/index.js": function(module, exports, __webpack_require__){
__webpack_require__("jquery")
__webpack_require__("lodash")
},
"jquery": function(module, exports){
module.exports = $;
},
"lodash": function(module, exports){
module.exports = _;
},
})这比较适用于一些第三方库来自于外部CDN的情况,这样一来,即可以在页面中使用CDN,又让bundle的体积变得更小,还不影响源码的编写
stats
stats控制的是构建过程中控制台的输出内容
好东西
https://tailwindcss.com/docs文档
https://play.tailwindcss.com/?ref=producthunt
<div class="w-full h-full bg-red-400 fixed"></div>
<!-- 本身不能设置高度,高度相对于父元素,变成绝对定位才可以 --><div class="w-full h-full flex bg-red-400 fixed">
<div class="w-60 bg-gray-800 h-full"></div>
<!-- <div class="w-60 bg-blue-200 h-full"></div> -->
<div class="bg-blue-200 h-full flex-grow"></div>
<!-- 宽度自动增长 -->
</div><!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta name="viewport" content="width=device-width, initial-scale=1.0" />
<title>Document</title>
</head>
<body>
<div class="w-full h-full flex bg-red-400 fixed">
<div class="w-60 bg-gray-800 h-full text-gray-50">
<h1 class="font-bold text-2xl text-center my-5">Logo</h1>
<ul>
<li class="text-center py-4 text-green-50 hover:text-green-300 hover:bg-gray-900"><a href="">Lorem.</a></li>
<li class="text-center py-4 text-green-50 hover:text-green-300 hover:bg-gray-900"><a href="">Earum.</a></li>
<li class="text-center py-4 text-green-50 hover:text-green-300 hover:bg-gray-900"><a href="">Eveniet?</a></li>
<li class="text-center py-4 text-green-50 hover:text-green-300 hover:bg-gray-900"><a href="">Veniam.</a></li>
<li class="text-center py-4 text-green-50 hover:text-green-300 hover:bg-gray-900"><a href="">Dolore!</a></li>
</ul>
</div>
<div class="bg-blue-200 h-full flex-grow"></div>
</div>
</body>
</html>配合postcss使用的
vscode插件:
智能提示:tailwind css intellisense
查看文档:tailwind docs
技术拓展
https://www.tailwindcss.cn/docs
https://github.com/yjisme/boilerplate-tranditional-proj
vscode插件:Tailwind css intellisense智能提示