JS 模組化 – 03 AMD 規範與 Require JS

1 AMD 規範介紹

AMD 規範,全稱 Asynchronous Module Definition,非同步模組定義,模組之間的依賴可以被非同步載入。

AMD 規範由 Common JS 規範演進而來,前文介紹 Common JS 規範時說過,瀏覽器端無法直接使用 Common JS,需要使用 browserify 編譯後才能運行。而 AMD 規範規範則專註於瀏覽器端。

1.1 定義模組

AMD 規範定義模組使用 define 函數,函數聲明如下:

define(id?, dependencies?, factory)

參數含義:

- id:非必填,模組的名字。如果沒有傳該參數,模組的名字默認為模組載入器請求的指定腳本的名字
- dependencies:非必填,數組,定義的這個模組所需要依賴的模組的數組。如果定義的這個模組不依賴於其他模組,則不需要傳遞該參數。
- factory:必填,工廠方法。如果為對象,則表示這個模組輸出的值;如果為函數,則是這個模組要乾的事。

如果傳遞了 dependenciesdependencies 中依賴的每項模組會在當前模組的 factory 之前執行。

1.2 載入模組

與 Common JS 規範類似,載入模組使用 require 函數,但 AMD 規範中該函數有兩個參數,語法格式如下:

require([module], callback)

參數含義:

- module:數組,要載入的模組數組。
- callback:模組載入成功之後要乾的事。

2 Require JS

require.js 是符合 AMD 規範的 JS 庫,使用 require.js 可以實現 AMD 規範,進行模組的定義和載入。

2.1 使用準備

首先下載 require.js 文件。require.js 可以從 github 下載。(不知道是不是網路原因,官網我打不開)

在項目中創建 lib 目錄,將 require.js 文件複製到 lib 目錄中。

2.2 初始化目錄

lib 同級目錄創建目錄 modules,在 modules 目錄中分別創建 module1.jsmodule2.js 代表兩個模組。

同時下載 moment.js 文件,將其複製到 lib 目錄中。

lib 同級目錄創建入口 HTML 和 JS 文件,名字分別為:index.htmlindex.js.

此時目錄結構為:

|- 03_AMD/
	|- lib/
		|- require.js
		|- moment.js
	|- modules/
		|- module1.js
		|- module2.js
	|- index.html
	|- index.js

index.html 文件中通過 script 標籤引入 require.js 文件,同時指定 data-main 屬性:

<script src="./lib/require.js" data-main="./index.js"></script>

data-main 屬性指定了在載入完 require.js 屬性後,執行的入口文件,該文件也稱為 主模組 。咱們的主模組為 index.html 同級路徑下的 index.js

2.3 路徑配置

在入口 JS 文件 index.js 中對模組進行配置:

requirejs.config({
  baseUrl: './',
  paths: {
    m1: './modules/module1',
    m2: './modules/module2',
    moment: './lib/moment'
  },
  shim: {
    moment: {
      exports: 'moment'
    },
  }
})
  1. baseUrl 屬性:指定了基本路徑,後面模組路徑配置都是相對於這個基本路徑。
  2. paths 屬性:配置各個模組的名稱及對應文件路徑(省略 .js 後綴)。上面分別給 module1.jsmodule2.jsmoment.js 三個模組命名為 m1m2moment。名字可以取其他名字,但路徑要正確。
  3. shim 屬性:當使用其他不符合 AMD 規範的模組時,可以使用該屬性導出模組。(這裡選擇的 moment.js 是兼容 AMD 規範的庫,無須配置到 shim 屬性中,此處僅為了簡單演示)

2.4 定義模組

前面創建了兩個自定義模組,現在分別編寫這兩個模組。

module1.js 中定義一個簡單的加法運算:

define(function () {
  console.log('in module1.')

  function sum(num1, num2) {
    console.log('module1 sum function.', num1, num2)
    return num1 + num2
  }

  return {
    sum
  }
})

module2.js 定義一個 calculate 函數,在該函數中需要調用 moment.js 中的格式化函數、 module1.js 中的 sum 函數,也就是說該模組(m2)依賴於 moment 模組 和 m1 模組:

define(['m1', 'moment'], function (m1, moment) {

  console.log('in module2.')

  function calculate (n1, n2) {
    console.log('begin calc: ', moment().format('YYYY MMM Do h:mm:ss a'))
    return m1.sum(n1, n2)
  }

  return {
    calculate
  }
})

2.5 使用模組

前面在主模組 index.js 中定義了模組的路徑,現在繼續在該文件中通過 require 函數使用其他模組:

// ...

require(['m2'], function (m2) {
  const result = m2.calculate(10, 20)
  console.log(result)
})

主模組中載入 m2 模組(module2.js),並調用 m2 模組中的 calculate 函數。

2.6 運行

在瀏覽器中運行 index.html,瀏覽器控制台輸出如下:

image-20220921224121047

主模組(index.js)依賴於 m2 模組(module2.js),m2 模組又依賴於 m1 模組(module1.js),故 require.js 會首先載入 module1.js,輸出 in module1.,然後載入 module2.js, 輸出 in module2.

m1m2moment 三個模組都載入完畢後,才會執行主模組中的 factory 工廠函數。

3 總結

AMD 規範的使用:

  • 定義模組:define 函數
  • 載入模組:require 函數

感謝你閱讀本文,如果本文給了你一點點幫助或者啟發,還請三連支援一下,點贊、關注、收藏,作者會持續與大家分享更多乾貨