摘要:入口模塊返回的賦值給總結(jié)在剖析了整體的流程之后,可以看到相關(guān)的技術(shù)細(xì)節(jié)還是比較清晰的,學(xué)無止境引用混合使用詳解的語法前端模塊化規(guī)范
前言
CMDAMD簡介
Commonjs簡介
Module簡介
Common和Module的區(qū)別
Module與webpack
Module與Babel
一些問題
總結(jié)
引用
前言前端模塊化在近幾年層出不窮,有Node的CommonJs,也有屬于client端的CMD/AMD模式,而ES6本身也出現(xiàn)了Modules,再加上Webpack以及babel的普及,雖然在代碼中經(jīng)常使用到這些用法,但是如果不去深入研究,總覺得是一個黑魔法,無法探測一些問題的根源。
AMD/CMD簡介事實上,隨著打包工具和Babel在前端工程化的世界里大放異彩,AMD/CMD也在逐步退出歷史的舞臺,這里簡單的介紹下其用法及語義。
AMD及其用法
AMD 即Asynchronous Module Definition,中文名是異步模塊定義的意思。代表(require.js)
/** main.js 入口文件/主模塊 **/ // 首先用config()指定各模塊路徑和引用名 require.config({ baseUrl: "js/lib", paths: { "jquery": "jquery.min", //實際路徑為js/lib/jquery.min.js "underscore": "underscore.min", } }); // 執(zhí)行基本操作 require(["jquery","underscore"],function($,_){ // some code here });
CMD及其用法
CMD 即Common Module Definition, 中文名是通用模塊定義的意思。代表(Sea.js)
/** sea.js **/ // 定義模塊 math.js define(function(require, exports, module) { var $ = require("jquery.js"); var add = function(a,b){ return a+b; } exports.add = add; }); // 加載模塊 seajs.use(["math.js"], function(math){ var sum = math.add(1+2); });
兩者的區(qū)別
1、AMD推崇依賴前置,在定義模塊的時候就要聲明其依賴的模塊
2、CMD推崇就近依賴,只有在用到某個模塊的時候再去require
Commonjs的應(yīng)用主要是在Node應(yīng)用中。
通過require引入文件, 文件內(nèi)部則通過module.export暴露,如下a 就是 module.export
// 引入某個文件 const a = require("some.js") // some.js module.export = { ... // some code }
除去module.export,Commonjs還有一個exports屬性(不推薦使用), 事實上exports就是module.export
// 對外輸出接口可以添加變量 var exports = module.exports; exports.area = function (r) { return Math.PI * r * r; }; exports.circumference = function (r) { return 2 * Math.PI * r; }; // 注意不要直接對exports賦值,這樣會切斷exports和module的關(guān)系 exports = a // 不要這么做Module簡介
ES6的Module是官方正式推出的模塊化寫法,雖然目前有挺多瀏覽器還不支持,不過我們可以利用babel將其轉(zhuǎn)換,話不多說,先介紹下Module的基本用法。
ES6的module主要是以import導(dǎo)入想要的對象,export 和 export default導(dǎo)出對象
import x from "some.js" // 引用some.js中的export default import {a, b} from "some.js" // 引用some.js的 export a 和 export b import x, {a, b} from "some.js" // 引用 some.js的 export default 和 export a 和 export b // some.js const x = () => {} export const a = () => {} export const b = () => {} export default x
因為import是編譯時加載,所以import命令具有提升效果,會提升到整個模塊的頭部,首先執(zhí)行。
// some code ... ... import xxx from "xxx" // 提升到最頂部Common和Module的區(qū)別
1. 加載的時機不同
Common是運行時加載的,可以使用變量或者表達(dá)式,如:
const "f" + "oo" = require("my_modules")
Module是編譯時加載的,不可以使用變量或者表達(dá)式, 編譯時加載效率較高。
2.暴露出的接口不同
Common暴露出來的是值的拷貝,也就是說,一旦輸出一個值,模塊內(nèi)部的變化就影響不到這個值。
// lib.js var counter = 3; function incCounter() { counter++; } module.exports = { counter: counter, incCounter: incCounter, };
// main.js var counter = require("./lib").counter; var incCounter = require("./lib").incCounter; console.log(counter); // 3 incCounter(); console.log(counter); // 3
Module則相反, 輸出的是值的引用。
Module與webpackwebpack本身維護了一套模塊系統(tǒng),這套模塊系統(tǒng)兼容了所有前端歷史進程下的模塊規(guī)范,包括 amd commonjs es6 等,為了看module在webpack中是怎么運行的,我們可以看一下下面簡單的代碼:
// webpack const path = require("path"); module.exports = { entry: "./a.js", output: { path: path.resolve(__dirname, "dist"), filename: "bundle.js", } };
// a.js import a from "./c"; export default "a.js"; console.log(a);
// c.js export default 333;
打包后的代碼如下:
(function(modules) { function __webpack_require__(moduleId) { var module = { i: moduleId, l: false, exports: {} }; modules[moduleId].call(module.exports, module, module.exports, __webpack_require__); return module.exports; } return __webpack_require__(0); })([ (function (module, __webpack_exports__, __webpack_require__) { // 引用 模塊 1 "use strict"; Object.defineProperty(__webpack_exports__, "__esModule", { value: true }); /* harmony import */ var __WEBPACK_IMPORTED_MODULE_0__c__ = __webpack_require__(1); /* harmony default export */ __webpack_exports__["default"] = ("a.js"); console.log(__WEBPACK_IMPORTED_MODULE_0__c__["a" /* default */]); }), (function (module, __webpack_exports__, __webpack_require__) { // 輸出本模塊的數(shù)據(jù) "use strict"; /* harmony default export */ __webpack_exports__["a"] = (333); }) ]);
簡化一波代碼再看,可以看出打包后實際上是一個立即執(zhí)行函數(shù),并且入?yún)楦鱾€module文件, 最后返回的是__webpack_require__(0):
(function(modules) { function __webpack_require__(moduleId) { } return __webpack_require__(0); })([module1, module2]);
ok, 我們繼續(xù)看__webpack_require__函數(shù),可以看出它是調(diào)用了我們的入口模塊,同時傳入了module相關(guān)的屬性,以及函數(shù)本身
function __webpack_require__(moduleId) { var module = { i: moduleId, l: false, exports: {} }; modules[moduleId].call(module.exports, module, module.exports, __webpack_require__); return module.exports; }
那么繼續(xù)追溯到入口模塊,也就是我們的第一個參數(shù)我們可以看到入口模塊又調(diào)用了 __webpack_require__(1) 去引用入?yún)?shù)組里的第2個函數(shù)。
然后會將入?yún)⒌?webpack_exports 對象添加 default 屬性,并賦值。
這里我們就能看到模塊化的實現(xiàn)原理,這里的 webpack_exports 就是這個模塊的 module.exports 通過對象的引用傳參,間接的給 module.exports 添加屬性。
最后會將 module.exports return 出來。就完成了 webpack_require 函數(shù)的使命。
function (module, __webpack_exports__, __webpack_require__) { /* harmony import */ var __WEBPACK_IMPORTED_MODULE_0__c__ = __webpack_require__(1); /* harmony default export */ __webpack_exports__["default"] = ("a.js"); console.log(__WEBPACK_IMPORTED_MODULE_0__c__["a" /* default */]); }
至此,我們可以看出module其實在webpack中,最后的打包結(jié)果。
Module與Babel雖然webpack可以打包轉(zhuǎn)換我們的module,但通常我們都會引入babel來對ES6轉(zhuǎn)成ES5的代碼,而Moduel屬于ES6,也會被轉(zhuǎn)譯。
事實上,babel是將module轉(zhuǎn)換成commonjs,這樣 webpack 就無需再做處理,直接使用 webpack 運行時定義的 webpack_require 處理。
不過babel在轉(zhuǎn)換的時候,會有一些特殊的處理, 像下面
首先 export 的時候, 會添加一個__esModule屬性到exports,是為了表明這是經(jīng)過轉(zhuǎn)換的module
export default a // 轉(zhuǎn)換成 Object.defineProperty(exports, "__esModule", { value: true }); exports.default = a;
再看 轉(zhuǎn)出的
轉(zhuǎn)出其實會多一個_interopRequireDefault函數(shù),就是為了處理default這個屬性
import d from "d" // 轉(zhuǎn)化后 var _d = require("d"); var _d2 = _interopRequireDefault(_d); function _interopRequireDefault(obj) { return obj && obj.__esModule ? obj : { default: obj }; }一些問題
1.為什么有的地方使用 require 去引用一個模塊時需要加上 default?
我們在上文 babel 對導(dǎo)出模塊的轉(zhuǎn)換提到,es6 的 export default 都會被轉(zhuǎn)換成 exports.default,即使這個模塊只有這一個輸出。
2.經(jīng)常在各大UI組件引用的文檔上會看到說明 import { button } from "xx-ui" 這樣會引入所有組件內(nèi)容,需要添加額外的 babel 配置,比如 babel-plugin-component?
import { Button, Select } from "element-ui" // 轉(zhuǎn)換成 var a = require("element-ui"); var Button = a.Button; var Select = a.Select;
babel-plugin-component就做了一件事,將 import { Button, Select } from "element-ui" 轉(zhuǎn)換成了
import Button from "element-ui/lib/button" import Select from "element-ui/lib/select"
3.我們在瀏覽一些 npm 下載下來的 UI 組件模塊時(比如說 element-ui 的 lib 文件下),看到的都是 webpack 編譯好的 js 文件,可以使用 import 或 require 再去引用。但是我們平時編譯好的 js 是無法再被其他模塊 import 的,這是為什么?
通過 webpack 模塊化原理章節(jié)給出的 webpack 配置編譯后的 js 是無法被其他模塊引用的,webpack 提供了 output.libraryTarget 配置指定構(gòu)建完的 js 的用途。入口模塊返回的 module.exports 賦值給 module.exports總結(jié)
在剖析了整體的流程之后,可以看到相關(guān)的技術(shù)細(xì)節(jié)還是比較清晰的,學(xué)無止境~~~
引用import、require、export、module.exports 混合使用詳解
Module的語法
前端模塊化
Commonjs規(guī)范
文章版權(quán)歸作者所有,未經(jīng)允許請勿轉(zhuǎn)載,若此文章存在違規(guī)行為,您可以聯(lián)系管理員刪除。
轉(zhuǎn)載請注明本文地址:http://m.specialneedsforspecialkids.com/yun/108907.html
摘要:系列文章系列第一篇基礎(chǔ)雜記系列第二篇插件機制雜記系列第三篇流程雜記前言公司的前端項目基本都是用來做工程化的,而雖然只是一個工具,但內(nèi)部涉及到非常多的知識,之前一直靠來解決問題,之知其然不知其所以然,希望這次能整理一下相關(guān)的知識點。 系列文章 Webpack系列-第一篇基礎(chǔ)雜記 Webpack系列-第二篇插件機制雜記 Webpack系列-第三篇流程雜記 前言 公司的前端項目基本都是用...
摘要:簡介是一種搭建客戶端的應(yīng)用架構(gòu),更像是一種模式而不是一個框架。 簡介 Flux是一種搭建WEB客戶端的應(yīng)用架構(gòu),更像是一種模式而不是一個框架。 特點 單向數(shù)據(jù)流 showImg(https://segmentfault.com/img/remote/1460000018128072?w=1300&h=708); 與MVC的比較 1.傳統(tǒng)的MVC如下所示(是一個雙向數(shù)...
摘要:最后執(zhí)行了的回調(diào)函數(shù),觸發(fā)了事件點,并回到函數(shù)的回調(diào)函數(shù)觸發(fā)了事件點執(zhí)行對于當(dāng)前模塊,或許存在著多個依賴模塊。 系列文章 Webpack系列-第一篇基礎(chǔ)雜記 Webpack系列-第二篇插件機制雜記 Webpack系列-第三篇流程雜記 前言 本文章個人理解, 只是為了理清webpack流程, 沒有關(guān)注內(nèi)部過多細(xì)節(jié), 如有錯誤, 請輕噴~ 調(diào)試 1.使用以下命令運行項目,./scrip...
摘要:系列文章系列第一篇基礎(chǔ)雜記系列第二篇插件機制雜記系列第三篇流程雜記前言本身并不難,他所完成的各種復(fù)雜炫酷的功能都依賴于他的插件機制。的插件機制依賴于一個核心的庫,。是什么是一個類似于的的庫主要是控制鉤子函數(shù)的發(fā)布與訂閱。 系列文章 Webpack系列-第一篇基礎(chǔ)雜記 Webpack系列-第二篇插件機制雜記 Webpack系列-第三篇流程雜記 前言 webpack本身并不難,他所完成...
摘要:前言自從發(fā)布之后,更新速度日新月異,而生命周期也隨之改變,雖然原有的一些生命周期函數(shù)面臨廢棄,但理解其背后更新的機制也是一種學(xué)習(xí)在這里根據(jù)官方文檔以及社區(qū)上其他優(yōu)秀的文章進行一個對于生命周期的總結(jié),大致上分為以下三個模塊新老生命周期的區(qū)別為 前言 自從React發(fā)布Fiber之后,更新速度日新月異,而生命周期也隨之改變,雖然原有的一些生命周期函數(shù)面臨廢棄,但理解其背后更新的機制也是一種...
閱讀 2118·2021-11-24 10:28
閱讀 1143·2021-10-12 10:12
閱讀 3350·2021-09-22 15:21
閱讀 691·2021-08-30 09:44
閱讀 1907·2021-07-23 11:20
閱讀 1155·2019-08-30 15:56
閱讀 1767·2019-08-30 15:44
閱讀 1490·2019-08-30 13:55