成人国产在线小视频_日韩寡妇人妻调教在线播放_色成人www永久在线观看_2018国产精品久久_亚洲欧美高清在线30p_亚洲少妇综合一区_黄色在线播放国产_亚洲另类技巧小说校园_国产主播xx日韩_a级毛片在线免费

資訊專欄INFORMATION COLUMN

vue源碼構(gòu)建代碼分析

RyanHoo / 2738人閱讀

這是xue源碼學(xué)習(xí)記錄,如有錯(cuò)誤請(qǐng)指出,謝謝!相互學(xué)習(xí)相互進(jìn)步。

vue源碼目錄為
vue
├── src        #vue源碼    
├── flow       #flow定義的數(shù)據(jù)類型庫(kù)(vue通過(guò)flow來(lái)檢測(cè)數(shù)據(jù)類型是否正確)         
├── examples   #demo   
├── scripts    #vue構(gòu)建命令
├── ...    

vue內(nèi)部代碼模塊比較清晰,這邊主要分析scripts內(nèi)部代碼,講解vue是如何進(jìn)行構(gòu)建的.
首先你必須要懂一些rollup,vue內(nèi)部是通過(guò)rollup來(lái)進(jìn)行構(gòu)建的,rollup是一款js的構(gòu)建工具,
將各個(gè)小模塊打包成一個(gè)總的模塊(只針對(duì)js文件,比較輕量,不會(huì)有css,img等壓縮,比較適合開發(fā)插件,
如果是ui組件庫(kù)的話,還是webpack構(gòu)建會(huì)比較好。)
rollup說(shuō)明文檔:https://rollupjs.cn/

文件主要是scripts下的alias.js,config.js和build.js三個(gè)文件組成

alias

主要就是提供文件對(duì)應(yīng)的路徑

const path = require("path")

const resolve = p => path.resolve(__dirname, "../", p)

// 以下是設(shè)置別名,與對(duì)應(yīng)的真實(shí)文件路徑
module.exports = {
  vue: resolve("src/platforms/web/entry-runtime-with-compiler"),
  compiler: resolve("src/compiler"),
  core: resolve("src/core"),
  shared: resolve("src/shared"),
  web: resolve("src/platforms/web"),
  weex: resolve("src/platforms/weex"),
  server: resolve("src/server"),
  entries: resolve("src/entries"),
  sfc: resolve("src/sfc"),
  test: resolve("src/test") // 這個(gè)是測(cè)試目錄是自己添加的
}

其中test是我自己加的,為了測(cè)試打包

config

config是為了提供打包的基礎(chǔ)配置(即rollup打包配置文件格式),由于打包內(nèi)容比較多,
所以做成可配置的

const path = require("path")
const buble = require("rollup-plugin-buble")
// 提供modules名稱的 alias 和reslove 功能
const alias = require("rollup-plugin-alias")
// 將CommonJS模塊轉(zhuǎn)換為 ES2015供Rollup 處理
const cjs = require("rollup-plugin-commonjs")
// 變量替換,可以將動(dòng)態(tài)設(shè)置的變量提取出來(lái)在配置文件中設(shè)置
const replace = require("rollup-plugin-replace")
// 幫助 Rollup 查找外部模塊,然后安裝
const node = require("rollup-plugin-node-resolve") 
const flow = require("rollup-plugin-flow-no-whitespace")
const version = process.env.VERSION || require("../package.json").version
// 下面是weex服務(wù)器端代碼,不需要管
// const weexVersion = process.env.WEEX_VERSION || require("../packages/weex-vue-framework/package.json").version

// 這邊是打包完成后模塊外部首行注釋代碼
const banner = 
  `/*
   * test-vue.js v${version}
   * (C) 2014-${new Date().getFullYear()} Enan You
   * @author zhengjie
   */
  `
// 獲取文件夾路徑別名
const aliases = require("./alias")
// 尋找路徑
const resolve = p => {
  const base = p.split("/")[0]
  if (aliases[base]) {
    return path.resolve(aliases[base], p.slice(base.length + 1))
  } else {
    return path.resolve(__dirname, "../", p)
  }
}
// 設(shè)置打包參數(shù) 
// 這邊把原先vue打包文件去除,替換成知己的一個(gè)測(cè)試文件
const builds = {
  "test-cjs": {
    entry: resolve("test/main.js"),
    dest: resolve("dist/test-cjs.js"),
    format: "cjs", // csj格式 module.exports
    banner
  },
  "test-es": {
    entry: resolve("test/main.js"),
    dest: resolve("dist/test-es.js"),
    format: "es", // es格式 export default
    banner
  },
  "test-umd": {
    entry: resolve("test/main.js"),
    dest: resolve("dist/test-umd.js"),
    format: "umd", // 瀏覽器格式 return
    banner
  }
}

// 根據(jù)上面builds打包轉(zhuǎn)換成rollup打包格式
function getConfig(name) {
  const opts = builds[name]
  const config = {
    input: opts.entry,
    external: opts.external,
    plugins: [
      replace({
        __WEEX__: !!opts.weex,
        __WEEX_VERSION__: null,
        __VERSION__: version
      }),
      flow(),
      buble(),
      alias(Object.assign({}, aliases, opts.alias))
    ].concat(opts.plugins || []),
    output: {
      file: opts.dest,
      format: opts.format,
      banner: opts.banner,
      name: opts.moduleName || "Vue"
    }
  }
  // 如果是開發(fā)模式
  if (opts.env) {
    config.plugins.push(replace({
      "process.env.NODE_ENV": JSON.stringify(opts.env)
    }))
  }
  // 增加屬性
  Object.defineProperty(config, "_name", {
    enumerable: false,
    value: name
  })

  return config
}

if (process.env.TARGET) {
  module.exports = getConfig(process.env.TARGET)
} else {
  exports.getBuild = getConfig
  exports.getAllBuilds = () => Object.keys(builds).map(getConfig)
}

其中里面的builds已被替換成我自己的測(cè)試文件,用于測(cè)試打敗es,模塊化和瀏覽器的不同格式。

build

build文件就是根據(jù)配置文件進(jìn)行打包,打包模式分為全部打包,或者是可配置打包,
如果運(yùn)行npm run build,將會(huì)打包所有的配置,
而運(yùn)行npm run build "參數(shù)", 則根據(jù)參數(shù)配置進(jìn)行打包

const fs = require("fs")
const path = require("path")
const zlib = require("zlib")
const rollup = require("rollup")
const uglify = require("uglify-js")
// 檢測(cè)是否有dist文件
if (!fs.existsSync("dist")) {
  fs.mkdirSync("dist")
}

let builds = require("./config").getAllBuilds()

// build后面輸入的參數(shù)
if (process.argv[2]) {
  // 過(guò)濾出需要打包的數(shù)組
  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 {
  // 這邊過(guò)濾出weex,不需要管
  builds = builds.filter(b => {
    return b.output.file.indexOf("weex") === -1
  })
}
// 把需要打包的打包出來(lái)
build(builds)
// 打包函數(shù)
function build(builds) {
  let built = 0
  const total = builds.length
  const next = () => {
    buildEntry(builds[built]).then(() => {
      built++
      if (built < total) {
        next()
      }
    }).catch(logError)
  }
  next()
}
// 單個(gè)配置文件打包
function buildEntry(config) {
  const output = config.output
  const {file, banner} = output
  // 是否為壓縮文件
  const isProd = /min.js$/.test(file)
  return rollup.rollup(config)
    .then(bundle => bundle.generate(output))
    .then(({code}) => {
      // 壓縮我文件
      if (isProd) {
        var minified = (banner ? banner + "
" : "") + uglify.minify(code, {
          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)
      if (zip) {
        zlib.gzip(code, (err, zipped) => {
          if (err) return reject(err)
          report(" (gzipped: " + getSize(zipped) + ")")
        })
      } else {
        report()
      }
    })
  })
}
// 計(jì)算文件大小
function getSize (code) {
  return (code.length / 1024).toFixed(2) + "kb"
}
// 輸入錯(cuò)誤信息
function logError (e) {
  console.log(e)
}
function blue (str) {
  return "x1b[1mx1b[34m" + str + "x1b[39mx1b[22m"
}
測(cè)試文件

main.js

import foo from "./foo";
export default function () {
  console.log(foo)
}

foo.js

export default "hello rollup"

通過(guò)npm run build打包測(cè)試,會(huì)在dist文件打包出test-cjs.js,test-es.js,test-umd.js
文件內(nèi)容如下:
test-cjs.js

/*
   * test-vue.js v1.0.0
   * (C) 2014-2018 Enan You
   * @author zhengjie
   */
  
"use strict";

var foo = "hello rollup"

function main () {
  console.log(foo);
}

module.exports = main;

test-es.js

/*
   * test-vue.js v1.0.0
   * (C) 2014-2018 Enan You
   * @author zhengjie
   */
  
var foo = "hello rollup"

function main () {
  console.log(foo);
}

export default main;

test-umd.js

/*
   * test-vue.js v1.0.0
   * (C) 2014-2018 Enan You
   * @author zhengjie
   */
  
(function (global, factory) {
    typeof exports === "object" && typeof module !== "undefined" ? module.exports = factory() :
    typeof define === "function" && define.amd ? define(factory) :
    (global.Vue = factory());
}(this, (function () { "use strict";

var foo = "hello rollup"

function main () {
  console.log(foo);
}

return main;

})));

這個(gè)構(gòu)建的過(guò)程不難,比起webpack的配置文件要容易懂很多
懂得了vue的構(gòu)建,接下來(lái)就可以開始vue源碼的學(xué)習(xí)了

文章版權(quán)歸作者所有,未經(jīng)允許請(qǐng)勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。

轉(zhuǎn)載請(qǐng)注明本文地址:http://systransis.cn/yun/98401.html

相關(guān)文章

  • Vue2 transition源碼分析

    摘要:至此算是找到了源碼位置。至此進(jìn)入過(guò)渡的部分完畢。在動(dòng)畫結(jié)束后,調(diào)用了由組件生命周期傳入的方法,把這個(gè)元素的副本移出了文檔流。這篇并沒(méi)有去分析相關(guān)的內(nèi)容,推薦一篇講非常不錯(cuò)的文章,對(duì)構(gòu)造函數(shù)如何來(lái)的感興趣的同學(xué)可以看這里 Vue transition源碼分析 本來(lái)打算自己造一個(gè)transition的輪子,所以決定先看看源碼,理清思路。Vue的transition組件提供了一系列鉤子函數(shù),...

    Genng 評(píng)論0 收藏0
  • vue 源碼學(xué)習(xí)(一) 目錄結(jié)構(gòu)和構(gòu)建過(guò)程簡(jiǎn)介

    摘要:框架使用了作為類型檢查,來(lái)保證項(xiàng)目的可讀性和維護(hù)性。的主目錄下有的配置文件,還有目錄,指定了各種自定義類型。表示構(gòu)建出來(lái)的文件遵循規(guī)范。 Flow vue框架使用了Flow作為類型檢查,來(lái)保證項(xiàng)目的可讀性和維護(hù)性。vue.js的主目錄下有Flow的配置.flowconfig文件,還有flow目錄,指定了各種自定義類型。 在學(xué)習(xí)源碼前可以先看下Flow的語(yǔ)法 官方文檔 目錄結(jié)構(gòu) vue....

    lavnFan 評(píng)論0 收藏0
  • VUE源碼--目錄結(jié)構(gòu)(一)

    摘要:源碼目錄結(jié)構(gòu)打包相關(guān)的配置文件,其中最重要的是。主要是根據(jù)不同的入口,打包為不同的文件。這個(gè)目錄下的代碼邏輯會(huì)把文件內(nèi)容解析成一個(gè)的對(duì)象。 源碼目錄結(jié)構(gòu) VUE 2.6.10 ├── scripts # 打包相關(guān)的配置文件,其中最重要的是config.js。主要是根據(jù)不同的入口,打 包為不同的文件。 ├── dist # 打包之后文...

    tuniutech 評(píng)論0 收藏0
  • Vue源碼解讀前必須要知道的Vue構(gòu)建流程

    摘要:了解的構(gòu)建,可以從文件開始命令內(nèi)關(guān)于構(gòu)建的命令如下版本從命令看出是通過(guò)來(lái)進(jìn)行構(gòu)建的同樣的構(gòu)建工具更強(qiáng)大一些,可以處理圖片等但是只做的處理,相比之下更輕量所以更適合框架的打包總共有三個(gè)版本的構(gòu)建從版本的為起點(diǎn)來(lái)了解,版本的構(gòu)建 了解Vue的構(gòu)建,可以從package.json文件開始sprits命令內(nèi)關(guān)于構(gòu)建的命令如下 { ... build: node scri...

    MASAILA 評(píng)論0 收藏0
  • Vue源碼之目錄結(jié)構(gòu)

    摘要:運(yùn)行時(shí)用來(lái)創(chuàng)建實(shí)例渲染并處理虛擬等的代碼?;旧暇褪浅ゾ幾g器的其它一切。版本可以通過(guò)標(biāo)簽直接用在瀏覽器中。為這些打包工具提供的默認(rèn)文件是只有運(yùn)行時(shí)的構(gòu)建。為瀏覽器提供的用于在現(xiàn)代瀏覽器中通過(guò)直接導(dǎo)入。 Vue版本:2.6.9 源碼結(jié)構(gòu)圖 ├─ .circleci // 包含CircleCI持續(xù)集成/持續(xù)部署工具的配置文件 ├─ .github ...

    freewolf 評(píng)論0 收藏0

發(fā)表評(píng)論

0條評(píng)論

最新活動(dòng)
閱讀需要支付1元查看
<