2.vuejs构建过程分析

Vue.js 源码基于Rollup构建。 Rollup较轻量。只编译js的部分,不支持编译图片和css等

先看下vuejs的package.json

// 地址 vue-2.6/package.json
"scripts": {
    // ...
    "build": "node scripts/build.js",
    "build:ssr": "npm run build -- web-runtime-cjs,web-server-renderer",
    "build:weex": "npm run build -- weex",
    // ...
},

这里总共有三条构建命令,vue-ssr vue-weex的版本在这里先不做分析。

web版本的构建过程分为五步

  1. 创建dist目录,后面打包好的文件都输出在dist目录下
  2. 拿到所有的build配置
  3. 根据传入的参数,对build配置进行过滤,排除不必要的打包项目
  4. 根据build的参数,生成打包后的代码
  5. 如果构建的是线上模式,则进行gzip压缩测试

1.创建dist目录

// 地址 vue-2.6/script/build
const fs = require('fs')
// 创建dist 
if (!fs.existsSync('dist')) {
  fs.mkdirSync('dist')
}

2.拿到所有的build配置

// 地址 vue-2.6/script/build
let builds = require('./config').getAllBuilds()

./config中的 getAllBuilds其实就是把默认的配置项转成Rollup需要的格式

// 地址 vue-2.6/script/config

/* builds的配置
 * entry:   入口路径,最终都指向了src/web目录下
 * dest:    输出的路径 最终都在dist目录下输出
 * format:  输出的规范 cjs => CommonJS,es => ES Module,umd => UMD。
 * env:     开发模式和线上模式  开发模式不压缩代码并且生成的.map文件会更详细
 * banner:  拼接在生成后的代码头部
 */
const builds = {
  // Runtime only (CommonJS). Used by bundlers e.g. Webpack & Browserify
  'web-runtime-cjs-dev': {
    entry: resolve('web/entry-runtime.js'),
    dest: resolve('dist/vue.runtime.common.dev.js'),
    format: 'cjs',
    env: 'development',
    banner
  }
  // 省略...
}
// 将builds处理成Rollup需要的格式
function genConfig(name) {
    const opts = builds[name]
    const config = {
    input: opts.entry,
    external: opts.external,
    plugins: [
      replace({
        __WEEX__: !!opts.weex,
        __WEEX_VERSION__: weexVersion,
        __VERSION__: version
      }),
      flow(),
      alias(Object.assign({}, aliases, opts.alias))
    ].concat(opts.plugins || []),
    output: {
      file: opts.dest,
      format: opts.format,
      banner: opts.banner,
      name: opts.moduleName || 'Vue'
    },
    onwarn: (msg, warn) => {
      if (!/Circular/.test(msg)) {
        warn(msg)
      }
    }
    }
    
    if (opts.env) {
    config.plugins.push(replace({
      'process.env.NODE_ENV': JSON.stringify(opts.env)
    }))
    }
    
    if (opts.transpile !== false) {
    config.plugins.push(buble())
    }
    
    Object.defineProperty(config, '_name', {
    enumerable: false,
    value: name
    })
    
    return config
}
exports.getAllBuilds = () => Object.keys(builds).map(genConfig)

排除不必要的打包项目

// npm run build:ssr process.argv[2] = web-runtime-cjs,web-server-renderer 这两种模式的区别下面会讲到
// npm run build:weex process.argv[2] = weex
// npm run build undefined
if (process.argv[2]) {
  const filters = process.argv[2].split(',')
  builds = builds.filter(b => {
    return filters.some(f => b.output.file.indexOf(f) > -1 || b._name.indexOf(f) > -1)
  })
} else {
  // filter out weex builds by default
  builds = builds.filter(b => {
    return b.output.file.indexOf('weex') === -1
  })
}

根据build的参数,生成打包后的代码

先来看看build方法

build(builds)

function build (builds) {
  let built = -100
  const total = builds.length
  const next = () => {
    buildEntry(builds[built]).then(() => {
      built++
      if (built < total) {
        next()
      }
    }).catch(logError)
  }

  next()
}
function logError (e) {
  console.log(e)
}

读到这里的时候产生了两个疑问。
1、为什么不直接用for循环调用buildEntry方法呢

// 为什么不这样实现呢? 因为:catch方法中没法办结束循环,会导致后面继续运行
for(let i = 0; i < builds.length; i++) {
   buildEntry(builds[i]).catch(logError)
}

2、logError是一个函数,可以不打括号这样调用吗

// 平时我习惯的声明函数方法是
let logError = e => {console.log(e)}
// 如果这样写就必须要 .catch((e) => logError(e)) 才能调到
// 但是如果是用function 声明的函数则不需要打括号传参过去

完整的构建函数

function build (builds) {
  let built = -100
  const total = builds.length
  const next = () => {
    buildEntry(builds[built]).then(() => {
      built++
      if (built < total) {
        next()
      }
    }).catch(logError)
  }

  next()
}

function buildEntry (config) {
  const output = config.output
  const { file, banner } = output
  // 是否需要压缩
  const isProd = /(min|prod)\.js$/.test(file)
  return rollup.rollup(config)
    .then(bundle => bundle.generate(output))
    .then(({ output: [{ code }] }) => {
      if (isProd) {
        // 处理压缩 和 map文件
        const minified = (banner ? banner + '\n' : '') + terser.minify(code, {
          toplevel: true,
          output: {
            ascii_only: true
          },
          compress: {
            pure_funcs: ['makeMap']
          }
        }).code
        return write(file, minified, true)
      } else {
        return write(file, code)
      }
    })
}

function write (dest, code, zip) {
  return new Promise((resolve, reject) => {
    function report (extra) {
      console.log(blue(path.relative(process.cwd(), dest)) + ' ' + getSize(code) + (extra || ''))
      resolve()
    }

    fs.writeFile(dest, code, err => {
      if (err) return reject(err)
      // gzip压缩
      if (zip) {
        zlib.gzip(code, (err, zipped) => {
          if (err) return reject(err)
          report(' (gzipped: ' + getSize(zipped) + ')')
        })
      } else {
        report()
      }
    })
  })
}

function getSize (code) {
  return (code.length / 1024).toFixed(2) + 'kb'
}

function logError (e) {
  console.log(e)
}

function blue (str) {
  return '\x1b[1m\x1b[34m' + str + '\x1b[39m\x1b[22m'
}

Runtime Only 和 Runtime+Compiler

在构建web版本的vue时,会默认构建出Runtime Only 和 Runtime+Compiler两个版本
1、Runtime Only

// 只认识render函数  不支持template: '<div>{{ hi }}</div>',更不认识.vue文件
new Vue({
  render (h) {
    return h('div', this.hi)
  }
})

2、 Runtime+Compiler

// 认识render函数和template,体积较大,编译template的过程耗时 
// 如果写template属性,则需要编译成render函数。这个编译过程会发生运行时
newew Vue({
  template: '<div>{{ hi }}</div>'
})

new Vue({
  render (h) {
    return h('div', this.hi)
  }
})

用脚手架构建的项目其实是通过vue-loader插件将.vue文件转为了render函数
所以更推荐Runtime Only
至于单文件的vue项目,如果你能不使用template,只用render函数写组件 那也更推荐Runtime Only

end...

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 216,258评论 6 498
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 92,335评论 3 392
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 162,225评论 0 353
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 58,126评论 1 292
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 67,140评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 51,098评论 1 295
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 40,018评论 3 417
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,857评论 0 273
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 45,298评论 1 310
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,518评论 2 332
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,678评论 1 348
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,400评论 5 343
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,993评论 3 325
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,638评论 0 22
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,801评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,661评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,558评论 2 352

推荐阅读更多精彩内容

  • 在分析Vue的源码之前我们需要了解一些前置知识,如Flow、源码目录、构建方式、编译入口等。 认识 Flow Fl...
    oWSQo阅读 1,081评论 1 2
  • vue.js 源码是基于 Rollup构建的,它的构建相关配置都放在script目录下。 构建脚本 通常一个基于N...
    LoveBugs_King阅读 572评论 0 1
  • 本文涉及包版本:node 11.6.0 、npm 6.11.3、webpack 4.39.3;使用mac开发; 项...
    前端gogogo阅读 7,743评论 2 6
  • 通过学习项目是如何build的,会有利于我们更好的了解整个项目的代码结构。以便于更好的学习项目的各个模块。打开pa...
    小A家的铭阅读 368评论 0 0
  • 距离分手还有76天, 一晚上迷迷糊糊的,我梦到你了,你说不会梦到我掉到河里去了吧,哈哈哈。今天,平邑下了初雪,下的...
    七控阅读 266评论 0 0