背景

在vue项目中,我们是将整个项目打包到一个dist目录下,但是在实际开发中,当项目足够庞大时(一级路由几十个,二级路由上百个),我们仅仅改一个小的BUG,比如说改了某个地方的一个按钮边框颜色,但是要发到线上的时候就需要打包整个项目,这样就会造成开发2分钟,打包半小时的现象,所以就需要拆分项目根据路由打包。

目的

如下所示项目结构:



根据views下面的模块(也就是路由),打包成各自的html,如下图



步骤

第一步

在package.json中添加以下代码

"build:all": "npm run build:healthWorksite && npm run build:inquiryManage && npm run build:remoteInquiry && npm run build:statistics && npm run build:staffWorkbench && npm run build:storage",
    "build:healthWorksite": "cross-env NODE_ENV=production env_config=prod route=healthWorksite node build/build.js",
    "build:inquiryManage": "cross-env NODE_ENV=production env_config=prod route=inquiryManage node build/build.js",
    "build:remoteInquiry": "cross-env NODE_ENV=production env_config=prod route=remoteInquiry node build/build.js",
    "build:statistics": "cross-env NODE_ENV=production env_config=prod route=statistics node build/build.js",
    "build:staffWorkbench": "cross-env NODE_ENV=production env_config=prod route=staffWorkbench node build/build.js",
    "build:storage": "cross-env NODE_ENV=production env_config=prod route=storage node build/build.js",
复制代码

其中

build:all是首次打包时,或者打包多个模块时使用,可以一次性按照顺序执行其中的指令;

cross-env是在打包之前配置打包变量的插件,这里需要设置route变量为路由名(路由名与文件夹名相同)

第二步

在view目录下的各自模块文件夹中,建立与模块文件名相同的html和js,分别作为各自的入口页面和入口js



html相当于原来src下面的index.html

js相当于原来src下面main.js

在js中,通用的依赖与原来的main.js相同,再添加各自页面需要的依赖,并且将router的指向改变为router目录下建立的自己模块路由,由于引用的App.vue没有做任何改变,所以引用外层的通用App.vue

示例代码如下:

import Vue from "vue";

import "@src/common";
import "@src/components";
import "normalize.css/normalize.css"; 
import "@src/assets/styles/index.scss"; 
import "@src/assets/styles/public.scss";
import App from "@src/App";
// 其他地方所有页面大体相同,这里引用的router不同
import router from "@src/router/healthWorksite/index.js"; 
import store from "@src/store";

// 以下代码省略
......

复制代码

第三步

在router目录下,建立与模块名相同的路由文件夹,并在文件夹下建立index.js,相当于原先router下面的index.js



在index.js中导入当前路由的配置,如上图,仅导入healthWorksit.js作为当前模块的路由

index.js示例代码如下:

import Vue from "vue";
import VueRouter from "vue-router";
import healthWorksite from "@src/router/routes/healthWorksite";
Vue.use(VueRouter);

const routes = [
  ...healthWorksite,
  {
    path: "/",
    redirect: {
      name: "/login"
    }
  }
];

const router = new VueRouter({
  base: "/",
  linkActiveClass: "active",
  routes
});
// 以下代码省略
......
复制代码

第四步

在config文件夹下的index.js里面添加以下代码

根据在package中设置的route变量去匹配打包后会使用的:



index-输出html文件名



assetsRoot-输出文件根目录路径



assetsSubDirectory-输出文件二级目录路径



assetsPublicPath-引用公共路径



用来生成不同路由下的打包配置,最后merge到输出项中

const merge = require("webpack-merge");
const buildNanme = process.env.route;
const buildList = {
  index: path.resolve(__dirname, `../../vueClinicWeb/${process.env.route}/index.html`),
  assetsRoot: path.resolve(__dirname, `../../vueClinicWeb/${process.env.route}`),
  assetsSubDirectory: "./static",
  assetsPublicPath: `../../vueClinicWeb/${process.env.route}`,
  productionSourceMap: false,
  devtool: "#source-map",
  productionGzip: false,
  productionGzipExtensions: ["js", "css"],
  bundleAnalyzerReport: process.env.npm_config_report
};
let buildRouteConfig = {};
buildRouteConfig[buildNanme] = buildList;
module.exports = merge(buildRouteConfig, {
  dev: {// 此处省略
  },

  build: merge(buildList, {
    index: path.resolve(__dirname, "../dist/index.html"),
    assetsRoot: path.resolve(__dirname, "../dist")
  })
});
复制代码

第五步

在build文件夹下面的utils.js中加入以下代码

根据路由匹配到上一步设置的路径,并输出该路径常量给其他js使用

// 根据路由进行配置
let routePathRoot = config.build.assetsRoot;
let routePathSubDirectory = config.build.assetsSubDirectory;
let routeAssetsPublicPath = config.build.assetsPublicPath;
let routeEntry = "./src/main.js";
let templatePath = "index.html";
let filename = config.build.index;
const ENV_ROUTE = process.env.route;
if (ENV_ROUTE) {
  routePathRoot = config[ENV_ROUTE].assetsRoot;
  routePathSubDirectory = config[ENV_ROUTE].assetsSubDirectory;
  routeAssetsPublicPath = config[ENV_ROUTE].assetsPublicPath;
  routeEntry = `./src/views/${ENV_ROUTE}/${ENV_ROUTE}.js`;
  templatePath = resolve(`/src/views/${ENV_ROUTE}/${ENV_ROUTE}.html`);
  filename = config[ENV_ROUTE].index;
}

exports.assetsPath = function(_path) {
  const assetsSubDirectory = process.env.NODE_ENV === "production" ? routePathSubDirectory : config.dev.assetsSubDirectory;

  return path.posix.join(assetsSubDirectory, _path);
};
exports.routePathRoot = routePathRoot;
exports.routePathSubDirectory = routePathSubDirectory;
exports.routeAssetsPublicPath = routeAssetsPublicPath;
exports.routeEntry = routeEntry;
exports.templatePath = templatePath;
exports.filename = filename;

复制代码

第六步

修改build文件夹下面的build.js

根据路由设置需要删除的文件夹路径

let routePathRoot = utils.routePathRoot;
let routePathSubDirectory = utils.routePathSubDirectory;

rm(path.join(routePathRoot, routePathSubDirectory)
// 以下代码省略
复制代码

第七步

修改build文件夹下的webpack.base.conf.js

根据路由设置通用的入口js文件路径、输出文件根目录、输出页面引用路径

// 仅显示修改部分
const routePathRoot = utils.routePathRoot;
const routeAssetsPublicPath = utils.routeAssetsPublicPath;
const routeEntry = utils.routeEntry;
module.exports = {
  entry: ["babel-polyfill", routeEntry],
  output: {
    path: routePathRoot,
    publicPath: process.env.NODE_ENV === "production" ? routeAssetsPublicPath : config.dev.assetsPublicPath
  }
};
复制代码

第八步

修改build文件夹下的webpack.prod.conf.js

根据路由设置文件名、入口页面路径、打包文件根目录路径、打包文件二级目录路径、引用资源公用路径等

// 仅展示修改部分
const filename = utils.filename;
const templatePath = utils.templatePath;
const routePathRoot = utils.routePathRoot;
const routePathSubDirectory = utils.routePathSubDirectory;
const routeAssetsPublicPath = utils.routeAssetsPublicPath;

const webpackConfig = merge(baseWebpackConfig, {
  output: {
    path: routePathRoot
  },
  plugins: [
    new HtmlWebpackPlugin({
      filename: filename,
      template: templatePath,
      inject: true,
      path: routeAssetsPublicPath + routePathSubDirectory,
      minify: {
        removeComments: true,
        collapseWhitespace: true,
        removeAttributeQuotes: true
      },
      chunksSortMode: "dependency"
    }),
    new CopyWebpackPlugin([
      {
        from: path.resolve(__dirname, "../static"),
        to: config.build.assetsSubDirectory,
        ignore: [".*"]
      }
    ])
  ]
});
复制代码

第九步

最后在终端中npm run build:all就大功告成了

首次打包推荐npm run build:all,之后再打包推荐改了哪个模块就打包那个模块的代码,这样就回到初衷,每次打包至少能节省几十倍的打包时间啦