很全很全的JavaScript的模塊講解(CommonJS,AMD,CMD,ES6模塊)

JavaScript的模塊

介紹

模塊通常是指編程語言所提供的代碼組織機制,利用此機制可將程序拆解爲獨立且通用的代碼單元。所謂模塊化主要是解決代碼分割、作用域隔離、模塊之間的依賴管理以及發佈到生產環境時的自動化打包與處理等多個方面。

模塊的優點

  1. 可維護性。 因爲模塊是獨立的,一個設計良好的模塊會讓外面的代碼對自己的依賴越少越好,這樣自己就可以獨立去更新和改進。
  2. 命名空間。 在 JavaScript 裏面,如果一個變量在最頂級的函數之外聲明,它就直接變成全局可用。因此,常常不小心出現命名衝突的情況。使用模塊化開發來封裝變量,可以避免污染全局環境。
  3. 重用代碼。 我們有時候會喜歡從之前寫過的項目中拷貝代碼到新的項目,這沒有問題,但是更好的方法是,通過模塊引用的方式,來避免重複的代碼庫。

CommonJS

CommonJS 最開始是 Mozilla 的工程師於 2009 年開始的一個項目,它的目的是讓瀏覽器之外的 JavaScript (比如服務器端或者桌面端)能夠通過模塊化的方式來開發和協作。

在 CommonJS 的規範中,每個 JavaScript 文件就是一個獨立的模塊上下文(module context),在這個上下文中默認創建的屬性都是私有的。也就是說,在一個文件定義的變量(還包括函數和類),都是私有的,對其他文件是不可見的。

需要注意的是,CommonJS 規範的主要適用場景是服務器端編程,所以採用同步加載模塊的策略。如果我們依賴3個模塊,代碼會一個一個依次加載它們。

該模塊實現方案主要包含 require 與 module 這兩個關鍵字,其允許某個模塊對外暴露部分接口並且由其他模塊導入使用。


 
  1. //sayModule.js

  2. function SayModule () {

  3. this.hello = function () {

  4. console.log('hello');

  5. };

  6.  
  7. this.goodbye = function () {

  8. console.log('goodbye');

  9. };

  10. }

  11.  
  12. module.exports = SayModule;

  13.  
  14. //main.js 引入sayModule.js

  15. var Say = require('./sayModule.js');

  16. var sayer = new Say();

  17. sayer.hello(); //hello

作爲一個服務器端的解決方案,CommonJS 需要一個兼容的腳本加載器作爲前提條件。該腳本加載器必須支持名爲 require 和 module.exports 的函數,它們將模塊相互導入導出。

Node.js

Node 從 CommonJS 的一些創意中,創造出自己的模塊化實現。由於Node 在服務端的流行,Node 的模塊形式被(不正確地)稱爲 CommonJS。

Node.js模塊可以分爲兩大類,一類是核心模塊,另一類是文件模塊。 
核心模塊 就是Node.js標準的API中提供的模塊,如fs、http、net等,這些都是由Node.js官方提供的模塊,編譯成了二進制代碼,可以直接通過require獲取核心模塊,例如require('fs'),核心模塊擁有最高的加載優先級,如果有模塊與核心模塊命名衝突,Node.js總是會加載核心模塊。 
文件模塊 是存儲爲單獨的文件(或文件夾)的模塊,可能是JavaScript代碼、JSON或編譯好的C/C++代碼。在不顯式指定文件模塊擴展名的時候,Node.js會分別試圖加上.js、.json、.node(編譯好的C/C++代碼)。

加載方式
  • 按路徑加載模塊

如果require參數一"/"開頭,那麼就以絕對路徑的方式查找模塊名稱,如果參數一"./"、"../"開頭,那麼則是以相對路徑的方式來查找模塊。

  • 通過查找node_modules目錄加載模塊

如果require參數不以"/"、"./"、"../"開頭,而該模塊又不是核心模塊,那麼就要通過查找node_modules加載模塊了。我們使用的npm獲取的包通常就是以這種方式加載的。

加載緩存

Node.js模塊不會被重複加載,這是因爲Node.js通過文件名緩存所有加載過的文件模塊,所以以後再訪問到時就不會重新加載了。 
注意: Node.js是根據實際文件名緩存的,而不是require()提供的參數緩存的,也就是說即使你分別通過require('express')和require('./node_modules/express')加載兩次,也不會重複加載,因爲儘管兩次參數不同,解析到的文件卻是同一個。

Node.js 中的模塊在加載之後是以單例化運行,並且遵循值傳遞原則:如果是一個對象,就相當於這個對象的引用。

模塊載入過程

加載文件模塊的工作,主要由原生模塊module來實現和完成,該原生模塊在啓動時已經被加載,進程直接調用到runMain靜態方法。


 
  1. 例如運行: node app.js

  2.  
  3. Module.runMain = function () {

  4. // Load the main module--the command line argument.

  5. Module._load(process.argv[1], null, true);

  6. };

  7.  
  8. //_load靜態方法在分析文件名之後執行

  9. var module = new Module(id, parent);

  10.  
  11. //並根據文件路徑緩存當前模塊對象,該模塊實例對象則根據文件名加載。

  12. module.load(filename);

具體說一下上文提到了文件模塊的三類模塊,這三類文件模塊以後綴來區分,Node.js會根據後綴名來決定加載方法,具體的加載方法在下文require.extensions中會介紹。

  • .js 通過fs模塊同步讀取js文件並編譯執行。
  • .node 通過C/C++進行編寫的Addon。通過dlopen方法進行加載。
  • .json 讀取文件,調用JSON.parse解析加載。

接下來詳細描述js後綴的編譯過程。Node.js在編譯js文件的過程中實際完成的步驟有對js文件內容進行頭尾包裝。以app.js爲例,包裝之後的app.js將會變成以下形式:


 
  1. //circle.js

  2. var PI = Math.PI;

  3. exports.area = function (r) {

  4. return PI * r * r;

  5. };

  6. exports.circumference = function (r) {

  7. return 2 * PI * r;

  8. };

  9.  
  10. //app.js

  11. var circle = require('./circle.js');

  12. console.log( 'The area of a circle of radius 4 is ' + circle.area(4));

  13.  
  14. //app包裝後

  15. (function (exports, require, module, __filename, __dirname) {

  16. var circle = require('./circle.js');

  17. console.log('The area of a circle of radius 4 is ' + circle.area(4));

  18. });

  19.  
  20. //這段代碼會通過vm原生模塊的runInThisContext方法執行(類似eval,只是具有明確上下文,不污染全局),返回爲一個具體的function對象。最後傳入module對象的exports,require方法,module,文件名,目錄名作爲實參並執行。

這就是爲什麼require並沒有定義在app.js 文件中,但是這個方法卻存在的原因。從Node.js的API文檔中可以看到還有__filename__dirnamemoduleexports幾個沒有定義但是卻存在的變量。其中__filename__dirname在查找文件路徑的過程中分析得到後傳入的。module變量是這個模塊對象自身,exports是在module的構造函數中初始化的一個空對象({},而不是null)。 
在這個主文件中,可以通過require方法去引入其餘的模塊。而其實這個require方法實際調用的就是module._load方法。 
load方法在載入、編譯、緩存了module後,返回module的exports對象。這就是circle.js文件中只有定義在exports對象上的方法才能被外部調用的原因。

以上所描述的模塊載入機制均定義在lib/module.js中。

require 函數

require 引入的對象主要是函數。當 Node 調用 require() 函數,並且傳遞一個文件路徑給它的時候,Node 會經歷如下幾個步驟:

  • Resolving:找到文件的絕對路徑;
  • Loading:判斷文件內容類型;
  • Wrapping:打包,給這個文件賦予一個私有作用範圍。這是使 require 和 module 模塊在本地引用的一種方法;
  • Evaluating:VM 對加載的代碼進行處理的地方;
  • Caching:當再次需要用這個文件的時候,不需要重複一遍上面步驟。
require.extensions 來查看對三種文件的支持情況


可以清晰地看到 Node 對每種擴展名所使用的函數及其操作:對 .js 文件使用 module._compile;對 .json 文件使用 JSON.parse;對 .node 文件使用 process.dlopen。

文件查找策略

  • 從文件模塊緩存中加載

儘管原生模塊與文件模塊的優先級不同,但是優先級最高的是從文件模塊的緩存中加載已經存在的模塊。

  • 從原生模塊加載

原生模塊的優先級僅次於文件模塊緩存的優先級。require方法在解析文件名之後,優先檢查模塊是否在原生模塊列表中。以http模塊爲例,儘管在目錄下存在一個httphttp.jshttp.nodehttp.json文件,require(“http”)都不會從這些文件中加載,而是從原生模塊中加載。 
原生模塊也有一個緩存區,同樣也是優先從緩存區加載。如果緩存區沒有被加載過,則調用原生模塊的加載方式進行加載和執行。

  • 從文件加載

當文件模塊緩存中不存在,而且不是原生模塊的時候,Node.js會解析require方法傳入的參數,並從文件系統中加載實際的文件,加載過程中的包裝和編譯細節在前面說過是調用load方法。 
··


 
  1. 當 Node 遇到 require(X) 時,按下面的順序處理。

  2.  
  3. (1)如果 X 是內置模塊(比如 require('http'))

  4.   a. 返回該模塊。

  5.   b. 不再繼續執行。

  6.  
  7. (2)如果 X 以 "./" 或者 "/" 或者 "../" 開頭

  8.   a. 根據 X 所在的父模塊,確定 X 的絕對路徑。

  9.   b. 將 X 當成文件,依次查找下面文件,只要其中有一個存在,就返回該文件,不再繼續執行。

  10. X

  11. X.js

  12. X.json

  13. X.node

  14.  
  15.   c. 將 X 當成目錄,依次查找下面文件,只要其中有一個存在,就返回該文件,不再繼續執行。

  16. X/package.json(main字段)

  17. X/index.js

  18. X/index.json

  19. X/index.node

  20.  
  21. (3)如果 X 不帶路徑

  22.   a. 根據 X 所在的父模塊,確定 X 可能的安裝目錄。

  23.   b. 依次在每個目錄中,將 X 當成文件名或目錄名加載。

  24.  
  25. (4) 拋出 "not found"

 

模塊循環依賴

 
  1. //創建兩個文件,module1.js 和 module2.js,並且讓它們相互引用

  2. // module1.js

  3. exports.a = 1;

  4. require('./module2');

  5. exports.b = 2;

  6. exports.c = 3;

  7.  
  8. // module2.js

  9. const Module1 = require('./module1');

  10. console.log('Module1 is partially loaded here', Module1);

在 module1 完全加載之前需要先加載 module2,而 module2 的加載又需要 module1。這種狀態下,我們從 exports 對象中能得到的就是在發生循環依賴之前的這部分。上面代碼中,只有 a 屬性被引入,因爲 b 和 c 都需要在引入 module2 之後才能加載進來。

Node 使這個問題簡單化,在一個模塊加載期間開始創建 exports 對象。如果它需要引入其他模塊,並且有循環依賴,那麼只能部分引入,也就是隻能引入發生循環依賴之前所定義的這部分。

AMD

AMD 是 Asynchronous Module Definition 的簡稱,即“異步模塊定義”,是從 CommonJS 討論中誕生的。AMD 優先照顧瀏覽器的模塊加載場景,使用了異步加載和回調的方式。

AMD 和 CommonJS 一樣需要腳本加載器,儘管 AMD 只需要對 define 方法的支持。define 方法需要三個參數:模塊名稱,模塊運行的依賴數組,所有依賴都可用之後執行的函數(該函數按照依賴聲明的順序,接收依賴作爲參數)。只有函數參數是必須的。define 既是一種引用模塊的方式,也是定義模塊的方式。


 
  1. // file lib/sayModule.js

  2. define(function (){

  3. return {

  4. sayHello: function () {

  5. console.log('hello');

  6. }

  7. };

  8. });

  9.  
  10. //file main.js

  11. define(['./lib/sayModule'], function (say){

  12. say.sayHello(); //hello

  13. })

main.js 作爲整個應用的入口模塊,我們使用 define 關鍵字聲明瞭該模塊以及外部依賴(沒有生命模塊名稱);當我們執行該模塊代碼時,也就是執行 define 函數的第二個參數中定義的函數功能,其會在框架將所有的其他依賴模塊加載完畢後被執行。這種延遲代碼執行的技術也就保證了依賴的併發加載。

RequireJS

RequireJS 是一個前端的模塊化管理的工具庫,遵循AMD規範,通過一個函數來將所有所需要的或者說所依賴的模塊實現裝載進來,然後返回一個新的函數(模塊),我們所有的關於新模塊的業務代碼都在這個函數內部操作,其內部也可無限制的使用已經加載進來的以來的模塊。


 
  1. <script data-main='scripts/main' src='scripts/require.js'></script>

  2. //scripts下的main.js則是指定的主代碼腳本文件,所有的依賴模塊代碼文件都將從該文件開始異步加載進入執行。

defined用於定義模塊,RequireJS要求每個模塊均放在獨立的文件之中。按照是否有依賴其他模塊的情況分爲獨立模塊和非獨立模塊。 
1、獨立模塊 不依賴其他模塊。直接定義


 
  1. define({

  2. methodOne: function (){},

  3. methodTwo: function (){}

  4. });

  5.  
  6. //等價於

  7.  
  8. define(function (){

  9. return {

  10. methodOne: function (){},

  11. methodTwo: function (){}

  12. };

  13. });

2、非獨立模塊,對其他模塊有依賴


 
  1. define([ 'moduleOne', 'moduleTwo' ], function(mOne, mTwo){

  2. ...

  3. });

  4.  
  5. //或者

  6.  
  7. define( function( require ){

  8. var mOne = require( 'moduleOne' ),

  9. mTwo = require( 'moduleTwo' );

  10. ...

  11. });

如上代碼, define中有依賴模塊數組的 和 沒有依賴模塊數組用require加載 這兩種定義模塊,調用模塊的方法合稱爲AMD模式,定義模塊清晰,不會污染全局變量,清楚的顯示依賴關係。AMD模式可以用於瀏覽器環境並且允許非同步加載模塊,也可以按需動態加載模塊。

CMD

CMD(Common Module Definition),在CMD中,一個模塊就是一個文件。

全局函數define,用來定義模塊。 
參數 factory 可以是一個函數,也可以爲對象或者字符串。 
當 factory 爲對象、字符串時,表示模塊的接口就是該對象、字符串。

定義JSON數據模塊:

define({ "foo": "bar" });

factory 爲函數的時候,表示模塊的構造方法,執行構造方法便可以得到模塊向外提供的接口。


 
  1. define( function(require, exports, module) {

  2. // 模塊代碼

  3. });

SeaJS

sea.js 核心特徵:

  1. 遵循CMD規範,與NodeJS般的書寫模塊代碼。
  2. 依賴自動加載,配置清晰簡潔。

seajs.use用來在頁面中加載一個或者多個模塊


 
  1.  
  2. // 加載一個模塊

  3. seajs.use('./a');

  4.  
  5. // 加載模塊,加載完成時執行回調

  6. seajs.use('./a',function(a){

  7. a.doSomething();

  8. });

  9.  
  10. // 加載多個模塊執行回調

  11. seajs.use(['./a','./b'],function(a , b){

  12. a.doSomething();

  13. b.doSomething();

  14. });

AMD和CMD最大的區別是對依賴模塊的執行時機處理不同,注意不是加載的時機或者方式不同。 
很多人說requireJS是異步加載模塊,SeaJS是同步加載模塊,這麼理解實際上是不準確的,其實加載模塊都是異步的,只不過AMD依賴前置,js可以方便知道依賴模塊是誰,立即加載,而CMD就近依賴,需要使用把模塊變爲字符串解析一遍才知道依賴了那些模塊,這也是很多人詬病CMD的一點,犧牲性能來帶來開發的便利性,實際上解析模塊用的時間短到可以忽略。

爲什麼說是執行時機處理不同? 
同樣都是異步加載模塊,AMD在加載模塊完成後就會執行該模塊,所有模塊都加載執行完後會進入回調函數,執行主邏輯,這樣的效果就是依賴模塊的執行順序和書寫順序不一定一致,看網絡速度,哪個先下載下來,哪個先執行,但是主邏輯一定在所有依賴加載完成後才執行。 
CMD加載完某個依賴模塊後並不執行,只是下載而已,在所有依賴模塊加載完成後進入主邏輯,遇到require語句的時候才執行對應的模塊,這樣模塊的執行順序和書寫順序是完全一致的。

UMD

統一模塊定義(UMD:Universal Module Definition )就是將 AMD 和 CommonJS 合在一起的一種嘗試,常見的做法是將CommonJS 語法包裹在兼容 AMD 的代碼中。


 
  1. (function(define) {

  2. define(function () {

  3. return {

  4. sayHello: function () {

  5. console.log('hello');

  6. }

  7. };

  8. });

  9. }(

  10. typeof module === 'object' && module.exports && typeof define !== 'function' ?

  11. function (factory) { module.exports = factory(); } :

  12. define

  13. ));

該模式的核心思想在於所謂的 IIFE(Immediately Invoked Function Expression),該函數會根據環境來判斷需要的參數類別

ES6模塊(module)

嚴格模式 

ES6 的模塊自動採用嚴格模式,不管有沒有在模塊頭部加上"use strict";。 
嚴格模式主要有以下限制。

  • 變量必須聲明後再使用
  • 函數的參數不能有同名屬性,否則報錯
  • 不能使用with語句
  • 不能對只讀屬性賦值,否則報錯
  • 不能使用前綴0表示八進制數,否則報錯
  • 不能刪除不可刪除的屬性,否則報錯
  • 不能刪除變量delete prop,會報錯,只能刪除屬性delete global[prop]
  • eval不會在它的外層作用域引入變量
  • eval和arguments不能被重新賦值
  • arguments不會自動反映函數參數的變化
  • 不能使用arguments.callee
  • 不能使用arguments.caller
  • 禁止this指向全局對象
  • 不能使用fn.caller和fn.arguments獲取函數調用的堆棧
  • 增加了保留字(比如protected、static和interface)

模塊Module

一個模塊,就是一個對其他模塊暴露自己的屬性或者方法的文件。

導出Export

作爲一個模塊,它可以選擇性地給其他模塊暴露(提供)自己的屬性和方法,供其他模塊使用。


 
  1. // profile.js

  2. export var firstName = 'qiqi';

  3. export var lastName = 'haobenben';

  4. export var year = 1992;

  5.  
  6. //等價於

  7.  
  8. var firstName = 'qiqi';

  9. var lastName = 'haobenben';

  10. var year = 1992;

  11. export {firstName, lastName, year}

1、 通常情況下,export輸出的變量就是本來的名字,但是可以使用as關鍵字重命名。


 
  1. function v1() { ... }

  2. function v2() { ... }

  3.  
  4. export {

  5. v1 as streamV1,

  6. v2 as streamV2,

  7. v2 as streamLatestVersion

  8. };

  9.  
  10. //上面代碼使用as關鍵字,重命名了函數v1和v2的對外接口。重命名後,v2可以用不同的名字輸出兩次。

2、 需要特別注意的是,export命令規定的是對外的接口,必須與模塊內部的變量建立一一對應關係。


 
  1. // 報錯

  2. export 1;

  3.  
  4. // 報錯

  5. var m = 1;

  6. export m;

  7.  
  8. //上面兩種寫法都會報錯,因爲沒有提供對外的接口。第一種寫法直接輸出1,第二種寫法通過變量m,還是直接輸出1。1只是一個值,不是接口。

  9.  
  10. / 寫法一

  11. export var m = 1;

  12.  
  13. // 寫法二

  14. var m = 1;

  15. export {m};

  16.  
  17. // 寫法三

  18. var n = 1;

  19. export {n as m};

  20.  
  21. //上面三種寫法都是正確的,規定了對外的接口m。其他腳本可以通過這個接口,取到值1。它們的實質是,在接口名與模塊內部變量之間,建立了一一對應的關係。

3、最後,export命令可以出現在模塊的任何位置,只要處於模塊頂層就可以。如果處於塊級作用域內,就會報錯,接下來說的import命令也是如此。


 
  1. function foo() {

  2. export default 'bar' // SyntaxError

  3. }

  4. foo()

導入import

作爲一個模塊,可以根據需要,引入其他模塊的提供的屬性或者方法,供自己模塊使用。

1、 import命令接受一對大括號,裏面指定要從其他模塊導入的變量名。大括號裏面的變量名,必須與被導入模塊(profile.js)對外接口的名稱相同。如果想爲輸入的變量重新取一個名字,import命令要使用as關鍵字,將輸入的變量重命名。

import { lastName as surename } from './profile';

2、import後面的from指定模塊文件的位置,可以是相對路徑,也可以是絕對路徑,.js路徑可以省略。如果只是模塊名,不帶有路徑,那麼必須有配置文件,告訴 JavaScript 引擎該模塊的位置。

3、注意,import命令具有提升效果,會提升到整個模塊的頭部,首先執行。


 
  1. foo();

  2.  
  3. import { foo } from 'my_module';

  4.  
  5. //上面的代碼不會報錯,因爲import的執行早於foo的調用。這種行爲的本質是,import命令是編譯階段執行的,在代碼運行之前。

4、由於import是靜態執行,所以不能使用表達式和變量,這些只有在運行時才能得到結果的語法結構。


 
  1. / 報錯

  2. import { 'f' + 'oo' } from 'my_module';

  3.  
  4. // 報錯

  5. let module = 'my_module';

  6. import { foo } from module;

  7.  
  8. // 報錯

  9. if (x === 1) {

  10. import { foo } from 'module1';

  11. } else {

  12. import { foo } from 'module2';

  13. }

5、最後,import語句會執行所加載的模塊,因此可以有下面的寫法。


 
  1. import 'lodash';

  2. //上面代碼僅僅執行lodash模塊,但是不輸入任何值。

默認導出(export default)

每個模塊支持我們導出一個沒有名字的變量,使用關鍵語句export default來實現.


 
  1. export default function(){

  2.         console.log("I am default Fn");

  3.    }

  4. //使用export default關鍵字對外導出一個匿名函數,導入這個模塊的時候,可以爲這個匿名函數取任意的名字

  5.  
  6. //取任意名字均可

  7. import sayDefault from "./module-B.js";

  8. sayDefault();

  9. //結果:I am default Fn

1、默認輸出和正常輸出的比較


 
  1. // 第一組

  2. export default function diff() { // 輸出

  3. // ...

  4. }

  5.  
  6. import diff from 'diff'; // 輸入

  7.  
  8. // 第二組

  9. export function diff() { // 輸出

  10. // ...

  11. };

  12.  
  13. import {diff} from 'diff'; // 輸入

  14.  
  15. //上面代碼的兩組寫法,第一組是使用export default時,對應的import語句不需要使用大括號;第二組是不使用export default時,對應的import語句需要使用大括號。

export default命令用於指定模塊的默認輸出。顯然,一個模塊只能有一個默認輸出,因此export default命令只能使用一次。所以,import命令後面纔不用加大括號,因爲只可能對應一個方法。

 

2、因爲export default本質是將該命令後面的值,賦給default變量以後再默認,所以直接將一個值寫在export default之後。


 
  1. / 正確

  2. export default 42;

  3.  
  4. // 報錯

  5. export 42;

  6.  
  7. //上面代碼中,後一句報錯是因爲沒有指定對外的接口,而前一句指定外對接口爲default。

3、如果想在一條import語句中,同時輸入默認方法和其他變量,可以寫成下面這樣。


 
  1. import _, { each } from 'lodash';

  2.  
  3. //對應上面代碼的export語句如下

  4. export default function (){

  5. //...

  6. }

  7. export function each (obj, iterator, context){

  8. //...

  9. }

export 與 import 的複合寫法

如果在一個模塊之中,先輸入後輸出同一個模塊,import語句可以與export語句寫在一起。


 
  1. export { foo, bar } from 'my_module';

  2.  
  3. // 等同於

  4. import { foo, bar } from 'my_module';

  5. export { foo, bar };

  6.  
  7. / 接口改名

  8. export { foo as myFoo } from 'my_module';

  9.  
  10. // 整體輸出

  11. export * from 'my_module';

注意事項 
1、聲明的變量,對外都是隻讀的。但是導出的是對象類型的值,就可修改。 
2、導入不存在的變量,值爲undefined。

ES6 中的循環引用

ES6 中,imports 是 exprts 的只讀視圖,直白一點就是,imports 都指向 exports 原本的數據,比如:


 
  1. //------ lib.js ------

  2. export let counter = 3;

  3. export function incCounter() {

  4. counter++;

  5. }

  6.  
  7. //------ main.js ------

  8. import { counter, incCounter } from './lib';

  9.  
  10. // The imported value `counter` is live

  11. console.log(counter); // 3

  12. incCounter();

  13. console.log(counter); // 4

  14.  
  15. // The imported value can’t be changed

  16. counter++; // TypeError

因此在 ES6 中處理循環引用特別簡單,看下面這段代碼:


 
  1. //------ a.js ------

  2. import {bar} from 'b'; // (1)

  3. export function foo() {

  4. bar(); // (2)

  5. }

  6.  
  7. //------ b.js ------

  8. import {foo} from 'a'; // (3)

  9. export function bar() {

  10. if (Math.random()) {

  11. foo(); // (4)

  12. }

  13. }

假設先加載模塊 a,在模塊 a 加載完成之後,bar 間接性地指向的是模塊 b 中的 bar。無論是加載完成的 imports 還是未完成的 imports,imports 和 exports 之間都有一個間接的聯繫,所以總是可以正常工作。

實例


 
  1. //---module-B.js文件---

  2. //導出變量:name

  3. export var name = "cfangxu";

  4.  
  5. moduleA模塊代碼:

  6. //導入 模塊B的屬性 name    

  7. import { name } from "./module-B.js";   

  8. console.log(name)

  9. //打印結果:cfangxu

批量導出


 
  1. //屬性name

  2. var name = "cfangxu";

  3. //屬性age

  4. var age  = 26;

  5. //方法 say

  6. var say = function(){

  7.        console.log("say hello");

  8.    }

  9. //批量導出

  10. export {name,age,say}

批量導入


 
  1. //導入 模塊B的屬性

  2. import { name,age,say } from "./module-B.js";

  3. console.log(name)

  4. //打印結果:cfangxu

  5. console.log(age)

  6. //打印結果:26

  7. say()

  8. //打印結果:say hello

重命名導入變量


 
  1. import {name as myName} from './module-B.js';

  2. console.log(myName) //cfangxu

整體導入


 
  1. /使用*實現整體導入

  2. import * as obj from "./module-B.js";

  3.  
  4. console.log(obj.name)

  5. //結果:"cfangxu"

  6. console.log(obj.age)

  7. //結果:26

  8. obj.say();

  9. //結果:say hello

推薦資料

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章