vue2 项目升级到vue3之后npm run build执行两遍打包
实际是在
@vue/cli-service升级到5.0版本之后出现的问题
先说解决方法
两种办法
- 执行
build的时候加一个--no-module
vue-cli-service build --no-module
- 修改
browserslist,一般在package.json中或者单独的.browserslistrc文件中,添加一个not ie 11
package.json
"browserslist": [
"> 1%",
"last 2 versions",
"not dead",
"not ie 11"
]
.browserslistrc
> 1%
last 2 versions
not dead
not ie 11
分析原因
通过执行 npm run build 的时候打印的日志可以发现两次打包之前都输出了不一样的日志
Building legacy bundle for production...
Building module bundle for production...
正常只执行一次的打包只会输出一种日志
Building for production...
然后我们根据日志输出的关键字在 @vue/cli-service 项目中查找一下,我们执行的是 build 命令,所以先看这个命令的文件 @vue/cli-service/lib/commands/build/index.js,搜索一下关键字 legacy bundle 会查找到第 116 行
if (args.target === 'app') {
const bundleTag = args.needsDifferentialLoading
? args.moduleBuild
? `module bundle `
: `legacy bundle `
: ``
logWithSpinner(`Building ${bundleTag}for ${mode}...`)
}
发现当 args.needsDifferentialLoading 为 true 的时候就会出现打包两次所出现的日志,所以基本可以肯定问题出在这个上,继续找一下它的复制,往上查找,在67行发现了赋值语句
args.needsDifferentialLoading = needsDifferentialLoading
继续查找 needsDifferentialLoading 变量声明和赋值的地方,往上看就可以看到
const { allProjectTargetsSupportModule } = require('../../util/targets')
let needsDifferentialLoading = args.target === 'app' && args.module
if (allProjectTargetsSupportModule) {
log(
`All browser targets in the browserslist configuration have supported ES module.\n` +
`Therefore we don't build two separate bundles for differential loading.\n`
)
needsDifferentialLoading = false
}
needsDifferentialLoading 初始值如果 args.module 是 false 的话就是 false
在正常的项目开发中
arr.target的值一定是app,如果开发的是插件的话,那么一般在打包的时候会指定--target lib
还有就是如果 allProjectTargetsSupportModule 这个值是true的话, needsDifferentialLoading 会被手动赋值成 false ,于是我们发现了两个可以让 needsDifferentialLoading 是 false 的方法
--no-module的原理
先查找 args.module 的复制,会发现没有直接的赋值,args是整个回调函数的参数,而且在下面还给 args中没有的部分值,附上了默认参数,第23行
api.registerCommand('build', {
description: 'build for production',
usage: 'vue-cli-service build [options] [entry|pattern]',
options: {
// ...
'--no-module': `build app without generating <script type="module"> chunks for modern browsers`,
// ...
}
}, async (args, rawArgs) => {
for (const key in defaults) {
if (args[key] == null) {
args[key] = defaults[key]
}
}
// ...
})
// defaults 第一行
const defaults = {
clean: true,
target: 'app',
module: true,
formats: 'commonjs,umd,umd-min'
}
可以看到 defaults 中给了 module 一个默认值true, 那怎么让 module 变成 false 呢,其实可以看到 options 中有一项 --no-module 的描述是: 构建应用程序,无需为现代浏览器生成< script type="module " >,到这里基本就能猜到了加个 --no-module 就可以把 module 赋值成 false 了,但猜到归猜到了,我们还是看一下具体的实现吧。
-
从
package.json中确定程序执行的入口bin/vue-cli-service.js"bin": { "vue-cli-service": "bin/vue-cli-service.js" }, -
在
bin/vue-cli-service.js中通过minimist解析了参数,并创建了Service的实例,并调用了run方法,并传入了解析后的参数minimist会把参数中以--no-开头的参数,解析为falseminimist/index.js
if (/^--no-.+/.test(arg)) { var key = arg.match(/^--no-(.+)/)[1]; setArg(key, false, arg); }bin/vue-cli-service.js
const Service = require('../lib/Service') const service = new Service(process.env.VUE_CLI_CONTEXT || process.cwd()) const rawArgv = process.argv.slice(2) const args = require('minimist')(rawArgv, {/*...*/}) const command = args._[0] service.run(command, args, rawArgv) -
Service在实例化的时候,添加了内置的plugin其中就包括了./command/build命令lib/Service.js
module.exports = class Service { constructor (context, { plugins, pkg, inlineOptions, useBuiltIn } = {}) { // ... this.commands = {} this.plugins = this.resolvePlugins(plugins, useBuiltIn) } resolvePlugins(inlinePlugins, useBuiltIn) { const idToPlugin = (id, absolutePath) => ({ id: id.replace(/^.//, 'built-in:'), apply: require(absolutePath || id) }) let plugins const builtInPlugins = [ './commands/build', // ... ].map((id) => idToPlugin(id)) if (inlinePlugins) { // ... } else { const projectPlugins = // ... plugins = builtInPlugins.concat(projectPlugins) } const orderedPlugins = sortPlugins(plugins) return orderedPlugins } } -
执行了
service.run方法,run方法中调用了init方法,在init方法中初始化好了插件之后,用传入的参数调用对应的回调函数async run (name, args = {}, rawArgv = []) { // load env variables, load user config, apply plugins await this.init(mode) args._ = args._ || [] let command = this.commands[name] if (!command || args.help || args.h) { command = this.commands.help } const { fn } = command return fn(args, rawArgv) } init() { // apply plugins. this.plugins.forEach(({ id, apply }) => { if (this.pluginsToSkip.has(id)) return apply(new PluginAPI(id, this), this.projectOptions) }) }为每一个插件创建了一个
PluginAPI的实例,PluginAPI提供了registerCommand方法,并把回调函数保存在了service.commands中class PluginAPI { constructor (id, service) { this.id = id this.service = service } registerCommand (name, opts, fn) { if (typeof opts === 'function') { fn = opts opts = null } this.service.commands[name] = { fn, opts: opts || {} } } } -
至此
build的回调函数就收到了解析后的参数module: false
not ie 11
来看第二种解决方案的原理,只要从 ../../util/targets 中导入的allProjectTargetsSupportModule 值为 true,就可以了
const { allProjectTargetsSupportModule } = require('../../util/targets')
if (allProjectTargetsSupportModule) {
needsDifferentialLoading = false
}
lib/util/targets.js
const projectTargets = getTargets()
const allModuleTargets = getTargets(
{ esmodules: true },
{ ignoreBrowserslistConfig: true }
)
const allProjectTargetsSupportModule = doAllTargetsSupportModule(projectTargets)
function doAllTargetsSupportModule (targets) {
const browserList = Object.keys(targets)
return browserList.every(browserName => {
if (!allModuleTargets[browserName]) {
return false
}
return semver.gte(
semver.coerce(targets[browserName]),
semver.coerce(allModuleTargets[browserName])
)
})
}
getTargets 是 babel 提供的方法,如果参数为空,返回 browserlists 查询的默认值,参考:babeljs.io/docs/en/bab…
传入 esmodules: true ,返回 github.com/babel/babel… 这个json文件中查询的结果.
在 doAllTargetsSupportModule 方法中对 browserList 和 allModuleTargets进行了比较,如果 browserList 中有 allModuleTargets 不存在的属性,就返回 false 或者 browserList 中的版本号,比 allModuleTargets 小,也会返回 false
输出对比一下这两个对象
// browserList
{
android: '98.0.0',
chrome: '97.0.0',
edge: '98.0.0',
firefox: '96.0.0',
ie: '11.0.0',
ios: '14.5.0',
opera: '82.0.0',
safari: '15.2.0',
samsung: '15.0.0'
}
// allModuleTargets
{
android: '61.0.0',
chrome: '61.0.0',
edge: '16.0.0',
firefox: '60.0.0',
ios: '10.3.0',
node: '13.2.0',
opera: '48.0.0',
safari: '10.1.0',
samsung: '8.2.0'
}
发现 browserList 比 allModuleTargets 中多了一个 ie: 11.0.0 ,那我们只要配置 browserlists 让他没有 ie 这一项就可以,ie最后的版本就是 11了,所以加一个 not IE 11 就可以了。
参考资料
\