欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

详解webpack 多页面/入口支持&公共组件单独打包

程序员文章站 2023-09-08 22:20:07
webpack系列目录 webpack 系列 五:webpack loaders 模块加载器 基于webpack搭建纯静态页面型...

webpack系列目录




webpack 系列 五:webpack loaders 模块加载器

基于webpack搭建纯静态页面型前端工程解决方案模板, 最终形态源码见github:

正文

本篇主要介绍:如何自动构建入口文件,并生成对应的output;公共js库如何单独打包。

多入口文件,自动扫描入口。同时支持spa和多页面型的项目

公共js库如何单独打包。

上一篇示例,主要介绍如何集成第三方js库到项目中使用,如jquery。示例的入口只有一个index,而且是将公共js库连同page.js一起打包到output.js中。那么在开发中会出现,每新增一个页面模块,就需要修改webpack.config.js配置文件(增加一个入口),而且如果用到的第三方库比较多,这样也容易导致jquery,react等代码库重复被合并到打包后的js,导致js体积过大,页面加载时间过长

基础结构和准备工作

以下示例基于上一篇进行改进,上一篇项目源码

目录结构说明

.
├── package.json    # 项目配置
├── src      # 源码目录
│ ├── pagea.html    # 入口文件a
│ ├── pageb.html    # 入口文件b
│ ├── css/     # css资源
│ ├── img/     # 图片资源
│ ├── js     # js&jsx资源
│ │ ├── pagea.js    # a页面入口
│ │ ├── pageb.js    # b页面入口
│ │ ├── lib/    # 没有存放在npm的第三方库或者下载存放到本地的基础库,如jquery、zepto、avalon
│ ├── pathmap.json   # 手动配置某些模块的路径,可以加快webpack的编译速度
├── webpack.config.js   # webpack配置入口

一:自动构建入口

官方多入口示例

webpack默认支持多入口,官方也有。配件文件webpack.config.js如下

//已简化
var path = require("path");
module.exports = {
 entry: {
  pagea: "./pagea",
  pageb: "./pageb"
 },
 output: {
  path: path.join(__dirname, "js"),
  filename: "[name].bundle.js",
  chunkfilename: "[id].chunk.js"
 }
}

每新增一个页面就需要在webpack.config.js的entry 中增加一个 pagec:"./pagec",页面少还好,页面一多,就有点麻烦了,而且配置文件,尽可能不改动。那么如何支持不修改配置呢?

自动构建入口函数

entry实际上是一个map对象,结构如下{filename:filepath},那么我们可以根据文件名匹配,很容易构造自动扫描器:
npm 中有一个用于文件名匹配的 glob模块,通过glob很容易遍历出src/js目录下的所有js文件:

安装glob模块

$ npm install glob --save-dev

修改webpack.config.js 配置,新增entries函数,修改entry:entries(),修改output的filename为"[name].js"

//引入glob
var glob = require('glob')
//entries函数
var entries= function () {
 var jsdir = path.resolve(srcdir, 'js')
 var entryfiles = glob.sync(jsdir + '/*.{js,jsx}')
 var map = {};

 for (var i = 0; i < entryfiles.length; i++) {
  var filepath = entryfiles[i];
  var filename = filepath.substring(filepath.lastindexof('\/') + 1, filepath.lastindexof('.'));
  map[filename] = filepath;
 }
 return map;
}
//修改入口,已经修改outp的filename
module.exports = {
 //entry: "./src/js/index.js",
 entry: entries(),
 output: {
  path: path.join(__dirname, "dist"),
  filename: "[name].js"
 },
 ......
 //以下省略,可以见下文详细配置

测试

1.在src/js目录中新增pagea.js

//js只有两行代码,在body中加一句话
var $ = require("jquery")
$("<div>这是jquery生成的多页面示例</div>").appendto("body")

2.新增pagea.html,也顺便修改原来的index.html 对于js文件名的更改

<!doctype html>
<html lang="en">
<head>
<meta charset="utf-8">
<title></title>
</head>
<body>
<script src="../dist/index.js"></script>
</body>
</html>

3.执行webpack,启动dev-server

$ webpack
$ webpack-dev-server

详解webpack 多页面/入口支持&公共组件单独打包

ok,成功打包生成pagea.js,成功运行

二:公共库单独打包

先来分析下,上个步骤打包的日志:

详解webpack 多页面/入口支持&公共组件单独打包

index.js 依赖了avalon 和 jquery,然后打包后的index.js 有480kb

pagea.js 只用了jquery,然后打包后的js 有294kb

那么如果引用的lib库多一点,又被很多页面引用,那么lib库就会被重复打包到page.js中去,模块越多重复加载的情况越严重。

如果把公共代码提取出来作为单独的js,那么就到处可以复用,浏览器也就可以进行缓存,这时候就需要用到webpack内置插件webpack.optimize.commonschunkplugin

commonschunkplugin 介绍

使用

new webpack.optimize.commonschunkplugin(options)

options
翻译得比较简单,详见:

  1. options.name or options.names(string|string[]): 公共模块的名称
  2. options.filename (string): 公开模块的文件名(生成的文件名)
  3. options.minchunks (number|infinity|function(module,count) - boolean): 为number表示需要被多少个entries依赖才会被打包到公共代码库;为infinity 仅仅创建公共组件块,不会把任何modules打包进去。并且提供function,以便于自定义逻辑。
  4. options.chunks(string[]):只对该chunks中的代码进行提取。
  5. options.children(boolean):如果为true,那么公共组件的所有子依赖都将被选择进来
  6. options.async(boolean|string):如果为true,将创建一个 option.name的子chunks(options.chunks的同级chunks) 异步common chunk
  7. options.minsize(number):所有公共module的size 要大于number,才会创建common chunk

2个常用的例子,更多例子见官方说明:

1.commons chunk for entries:针对入口文件提取公共代码

new commonschunkplugin({
 name: "commons",
 // (the commons chunk name)

 filename: "commons.js",
 // (the filename of the commons chunk)

 // minchunks: 3,
 // (modules must be shared between 3 entries)

 // chunks: ["pagea", "pageb"],
 // (only use these entries)
})

2.explicit vendor chunk:直接指定第三方依赖库,打包成公共组件

entry: {
 vendor: ["jquery", "other-lib"],
 app: "./entry"
}
new commonschunkplugin({
 name: "vendor",

 // filename: "vendor.js"
 // (give the chunk a different name)

 minchunks: infinity,
 // (with more entries, this ensures that no other module
 // goes into the vendor chunk)
})

commonschunkplugin使用

基于上篇的项目,参考上面的第二个例子,我们将jquery 和 avalon 提取出来打包成vendor.js

完整的webpack.config.js 如下:

```js 
var webpack = require("webpack");
 var path = require("path");
 var srcdir = path.resolve(process.cwd(), 'src');
 var nodemodpath = path.resolve(__dirname, './node_modules');
 var pathmap = require('./src/pathmap.json');
 var glob = require('glob')
 var commonschunkplugin = webpack.optimize.commonschunkplugin;
 var entries= function () {
 var jsdir = path.resolve(srcdir, 'js')
 var entryfiles = glob.sync(jsdir + '/*.{js,jsx}')
 var map = {};
 for (var i = 0; i < entryfiles.length; i++) {
  var filepath = entryfiles[i];
  var filename = filepath.substring(filepath.lastindexof('\/') + 1, filepath.lastindexof('.'));
  map[filename] = filepath;
 }
 return map;
}

module.exports = {
 //entry: "./src/js/index.js",
 //entry: entries(),
 entry: object.assign(entries(), {
  // 用到什么公共lib(例如jquery.js),就把它加进vendor去,目的是将公用库单独提取打包
  'vendor': ['jquery', 'avalon']
 }),
 output: {
  path: path.join(__dirname, "dist"),
  filename: "[name].js"
 },
 module: {
  loaders: [
   {test: /\.css$/, loader: 'style-loader!css-loader'}
  ]
 },
 resolve: {
  extensions: ['.js', "", ".css"],
  root: [srcdir,nodemodpath],
  alias: pathmap,
  publicpath: '/'
 },
 plugins: [
  new commonschunkplugin({
   name: 'vendor',
   minchunks: infinity
  })
 ]
}
```

测试、验证

1.修改入口(object.assign 是html5.js里面的....)

//entry: entries(),
entry: object.assign(entries(), {
 // 用到什么公共lib(例如jquery.js),就把它加进vendor去,目的是将公用库单独提取打包
 'vendor': ['jquery', 'avalon']
}),

2.加入插件commonschunkplugin

var commonschunkplugin = webpack.optimize.commonschunkplugin;
config 中增加 plugins
 plugins: [
  new commonschunkplugin({
   name: 'vendor',
   minchunks: infinity
  })
 ]

3.修改index.html 和 pagea.html,增加对verdor.js的引用

<script src="../dist/vendor.js"></script>
<script src="../dist/index.js"></script>
//<script src="../dist/pagea.js"></script>

4.执行webpack

$ webpack

详解webpack 多页面/入口支持&公共组件单独打包

结果分析

可以看到index.js 就只有457 bytes了,pagea.js 227bytes。vendor.js 是集成了jquery+avalon,所以有488kb。

这样vendor.js 就可以重复利用了,也方便浏览器进行缓存。

调试过程中发现

uncaught referenceerror: webpackjsonp is not defined

这个是因为当时把vendor.js引入 放到了page.js 后面,导致page.js执行异常,所以,请一定把vendor.js 放在前面。

生成后的index.js就很轻便了,第三方库都被打包到vendor中了,代码如下:

webpackjsonp([0],[
/* 0 */
/***/ function(module, exports, __webpack_require__) {

 /**
  * created by sloong on 2016/6/1.
  */
 //avalon 测试
 var avalon = __webpack_require__(1);
 avalon.define({
  $id: "avalonctrl",
  name: "hello avalon!"
 });

 /*
 //zepto 测试
 require("zepto")

 $("<div>这是zepto生成的</div>").appendto("body")*/

 //jquery 测试
 var $ = __webpack_require__(2)
 $("<div>这是jquery生成的</div>").appendto("body")

/***/ }
]);

页面测试均正常

详解webpack 多页面/入口支持&公共组件单独打包

ok,本篇结束了。如何让webpack 自动在html文件中引入所需js的script标签,如何给js和css文件加了hash值,这样浏览器每次都能检测到文件变更,而且也不需要手动修改引入的js文件链接,这些操作webpack都能轻松搞定

以上就是本文的全部内容,希望对大家的学习有所帮助,也希望大家多多支持。