自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

這些 ECMAScript 模塊知識(shí),都是我需要知道的

開(kāi)發(fā) 前端
ECMAScript模塊(簡(jiǎn)稱ES模塊)是一種JavaScript代碼重用的機(jī)制,于2015年推出,一經(jīng)推出就受到前端開(kāi)發(fā)者的喜愛(ài)。在2015之年,JavaScript 還沒(méi)有一個(gè)代碼重用的標(biāo)準(zhǔn)機(jī)制。多年來(lái),人們對(duì)這方面的規(guī)范進(jìn)行了很多嘗試,導(dǎo)致現(xiàn)在有多種模塊化的方式。

[[341136]]

ES 模塊是什么?

ECMAScript模塊(簡(jiǎn)稱ES模塊)是一種JavaScript代碼重用的機(jī)制,于2015年推出,一經(jīng)推出就受到前端開(kāi)發(fā)者的喜愛(ài)。在2015之年,JavaScript 還沒(méi)有一個(gè)代碼重用的標(biāo)準(zhǔn)機(jī)制。多年來(lái),人們對(duì)這方面的規(guī)范進(jìn)行了很多嘗試,導(dǎo)致現(xiàn)在有多種模塊化的方式。

你可能聽(tīng)說(shuō)過(guò)AMD模塊,UMD,或CommonJS,這些沒(méi)有孰優(yōu)孰劣。最后,在ECMAScript 2015中,ES 模塊出現(xiàn)了。

我們現(xiàn)在有了一個(gè)“正式的”模塊系統(tǒng)。

ES 模塊無(wú)處不在?

理論上,ES 模塊應(yīng)該在所有JavaScript環(huán)境中。實(shí)際上,ES 模塊的主要應(yīng)用還是在瀏覽器上。

2020年5月,Node.js v12.17.0 增加了在不使用標(biāo)記前提下對(duì)ECMAScript模塊的支持。這意味著我們現(xiàn)在可以在Node.js中使用import和export,而無(wú)需任何其他命令行標(biāo)志。

ECMAScript模塊要想在任何JavaScript環(huán)境通用,可能還需要很長(zhǎng)的路要走,但方向是正確的。

ES 模塊是什么樣的

ES 模塊是一個(gè)簡(jiǎn)單的文件,我們可以在其中聲明一個(gè)或多個(gè)導(dǎo)出。以下面utils.js為例:

  1. // utils.js 
  2. export function funcA() { 
  3.   return "Hello named export!"
  4.  
  5. export default function funcB() { 
  6.   return "Hello default export!"

這里有兩個(gè)導(dǎo)出。

第一個(gè)是命名導(dǎo)出,后面是export default,表示為默認(rèn)導(dǎo)出。

假設(shè)我們的項(xiàng)目文件夾中有一個(gè)名為utils.js的文件,我們可以將這個(gè)模塊提供的對(duì)象導(dǎo)入到另一個(gè)文件中。

如何從 ES模塊 導(dǎo)入

假設(shè)我們?cè)陧?xiàng)目文中還有一個(gè)Consumer.js的文件。要導(dǎo)入utils.js公開(kāi)的函數(shù),我們可以這樣做:

  1. // consumer.js 
  2. import { funcA } from "./util.js"

這種對(duì)應(yīng)我們的命名導(dǎo)入方式.

如果我們要導(dǎo)入 utils.js 中的默認(rèn)導(dǎo)出也就是 funcB 方法,我們可以這樣做:

  1. // consumer.js 
  2. import { funcA } from "./util.js"

當(dāng)然,我們可以導(dǎo)入同時(shí)導(dǎo)入命名和默認(rèn)的:

  1. // consumer.js 
  2. import funcB, { funcA } from "./util.js"
  3.  
  4. funcB(); 
  5. funcA(); 

我們也可以用星號(hào)導(dǎo)入整個(gè)模塊:

  1. import * as myModule from './util.js'
  2.  
  3. myModule.funcA(); 
  4. myModule.default(); 

注意,這里要使用默認(rèn)到處的方法是使用 default() 而不是 funcB()。

從遠(yuǎn)程模塊導(dǎo)入:

  1. import { createStore } from "https://unpkg.com/redux@4.0.5/es/redux.mjs"
  2.  
  3. const store = createStore(/* do stuff */) 

瀏覽器中的 ES 模塊

現(xiàn)代瀏覽器支持ES模塊,但有一些警告。要使用模塊,需要在 script 標(biāo)簽上添加屬性 type, 對(duì)應(yīng)值 為 module。

  1. <html lang="en"
  2. <head> 
  3.     <meta charset="UTF-8"
  4.     <title>ECMAScript modules in the browser</title> 
  5. </head> 
  6. <body> 
  7. <p id="el">The result is: </p> 
  8. </body> 
  9. <script type="module"
  10.     import { appendResult } from "./myModule.js"
  11.  
  12.     const el = document.getElementById("el"); 
  13.     appendResult(el); 
  14. </script> 
  15. </html> 

 myModule.js 內(nèi)容如下:

  1. export function appendResult(element) { 
  2.   const result = Math.random(); 
  3.   element.innerText += result; 

動(dòng)態(tài)導(dǎo)入

ES 模塊是靜態(tài)的,這意味著我們不能在運(yùn)行時(shí)更改導(dǎo)入。隨著2020年推出的動(dòng)態(tài)導(dǎo)入(dynamic imports),我們可以動(dòng)態(tài)加載代碼來(lái)響應(yīng)用戶交互(webpack早在ECMAScript 2020推出這個(gè)特性之前就提供了動(dòng)態(tài)導(dǎo)入)。

考慮下面的代碼:

  1. <!DOCTYPE html> 
  2. <html lang="en"
  3. <head> 
  4.     <meta charset="UTF-8"
  5.     <title>Dynamic imports</title> 
  6. </head> 
  7. <body> 
  8. <button id="btn">Load!</button> 
  9. </body> 
  10. <script src="loader.js"></script> 
  11. </html> 

再考慮一個(gè)帶有兩個(gè)導(dǎo)出的JavaScript模塊

  1. // util.js 
  2. export function funcA() { 
  3.   console.log("Hello named export!"); 
  4.  
  5. export default function funcB() { 
  6.   console.log("Hello default export!"); 

為了動(dòng)態(tài)導(dǎo)入 util.js 模塊,我們可以點(diǎn)擊按鈕在去導(dǎo)入:

  1. / loader.js 
  2. const btn = document.getElementById("btn"); 
  3.  
  4. btn.addEventListener("click", () => { 
  5.   // loads named export 
  6.   import("./util.js").then(({ funcA }) => { 
  7.     funcA(); 
  8.   }); 
  9. }); 

這里使用解構(gòu)的方式,取出**命名導(dǎo)出 ** funcA 方法:

  1. ({ funcA }) => {} 

ES模塊實(shí)際上是JavaScript對(duì)象:我們可以解構(gòu)它們的屬性以及調(diào)用它們的任何公開(kāi)方法。

要使用動(dòng)態(tài)導(dǎo)入的默認(rèn)方法,可以這樣做

  1. // loader.js 
  2. const btn = document.getElementById("btn"); 
  3.  
  4. btn.addEventListener("click", () => { 
  5.   import("./util.js").then((module) => { 
  6.     module.default(); 
  7.   }); 
  8. }); 

當(dāng)作為一個(gè)整體導(dǎo)入一個(gè)模塊時(shí),我們可以使用它的所有導(dǎo)出

  1. // loader.js 
  2. const btn = document.getElementById("btn"); 
  3.  
  4. btn.addEventListener("click", () => { 
  5.   // loads entire module 
  6.   // uses everything 
  7.   import("./util.js").then((module) => { 
  8.     module.funcA(); 
  9.     module.default(); 
  10.   }); 
  11. }); 

還有另一種用于動(dòng)態(tài)導(dǎo)入的常見(jiàn)樣式,如下所示:

  1. const loadUtil = () => import("./util.js"); 
  2.  
  3. const btn = document.getElementById("btn"); 
  4.  
  5. btn.addEventListener("click", () => { 
  6.   // 
  7. }); 

loadUtil返回的是一個(gè) promise,所以我們可以這樣操作

  1. const loadUtil = () => import("./util.js"); 
  2.  
  3. const btn = document.getElementById("btn"); 
  4.  
  5. btn.addEventListener("click", () => { 
  6.   loadUtil().then(module => { 
  7.     module.funcA(); 
  8.     module.default(); 
  9.   }) 
  10. }) 

動(dòng)態(tài)導(dǎo)入看起來(lái)不錯(cuò),但是它們有什么用呢?

使用動(dòng)態(tài)導(dǎo)入,我們可以拆分代碼,并只在適當(dāng)?shù)臅r(shí)候加載重要的代碼。在 JavaScript 引入動(dòng)態(tài)導(dǎo)入之前,這種模式是webpack(模塊綁定器)獨(dú)有的。

像React和Vue通過(guò)動(dòng)態(tài)導(dǎo)入代碼拆分來(lái)加載響應(yīng)事件的代碼塊,比如用戶交互或路由更改。

動(dòng)態(tài)導(dǎo)入JSON文件

假設(shè)我們項(xiàng)目有一個(gè) person.json 文件,內(nèi)容如下:

  1.   "name""Jules"
  2.   "age": 43 

現(xiàn)在,我們需要?jiǎng)討B(tài)導(dǎo)入該文件以響應(yīng)某些用戶交互。

因?yàn)?JSON 文件不是一個(gè)方法,所以我們可以使用默認(rèn)導(dǎo)出方式:

  1. const loadPerson = () => import('./person.json'); 
  2.  
  3. const btn = document.getElementById("btn"); 
  4.  
  5. btn.addEventListener("click", () => { 
  6.   loadPerson().then(module => { 
  7.     const { name, age } = module.default
  8.     console.log(name, age); 
  9.   }); 
  10. }); 

這里我們使用解構(gòu)的方式取出 name 和 age :

  1. const { name, age } = module.default

動(dòng)態(tài)導(dǎo)入與 async/await

因?yàn)?import() 語(yǔ)句返回是一個(gè) Promise,所以我們可以使用 async/await:

  1. const loadUtil = () => import("./util.js"); 
  2.  
  3. const btn = document.getElementById("btn"); 
  4.  
  5. btn.addEventListener("click", async () => { 
  6.   const utilsModule = await loadUtil(); 
  7.   utilsModule.funcA(); 
  8.   utilsModule.default(); 
  9. }) 

動(dòng)態(tài)導(dǎo)入的名字

使用import()導(dǎo)入模塊時(shí),可以按照自己的意愿命名它,但要調(diào)用的方法名保持一致:

  1. import("./util.js").then((module) => { 
  2.     module.funcA(); 
  3.     module.default(); 
  4.   }); 

或者:

  1. import("./util.js").then((utilModule) => { 
  2.   utilModule.funcA(); 
  3.   utilModule.default(); 
  4. }); 

作者:Valentino Gagliardib 譯者:前端小智 來(lái)源:valentinog

原文:https://www.valentinog.com/blog/es-modules/

本文轉(zhuǎn)載自微信公眾號(hào)「大遷世界」,可以通過(guò)以下二維碼關(guān)注。轉(zhuǎn)載本文請(qǐng)聯(lián)系大遷世界公眾號(hào)。

 

責(zé)任編輯:武曉燕 來(lái)源: 大遷世界
相關(guān)推薦

2018-02-08 08:08:12

2022-04-28 12:17:26

瀏覽器連字符hyphens

2018-05-16 09:41:13

神經(jīng)網(wǎng)絡(luò)NN函數(shù)

2016-01-20 09:44:22

物聯(lián)網(wǎng)標(biāo)準(zhǔn)

2018-01-03 11:35:34

推送AndroidiOS

2016-11-17 18:37:44

機(jī)房建設(shè)

2014-09-01 14:31:11

2018-04-27 09:58:51

2012-07-27 09:25:40

2017-10-02 10:39:48

2018-01-09 15:44:57

2015-07-15 10:26:29

2022-02-09 16:25:34

區(qū)塊鏈技術(shù)加密貨幣

2019-02-01 10:23:05

2021-05-08 11:16:44

手機(jī)快充技術(shù)

2018-07-03 14:40:31

2020-12-30 11:22:11

Node.js前端模塊

2011-12-13 10:16:34

2022-09-22 08:00:00

API開(kāi)發(fā)數(shù)據(jù)

2023-09-08 13:46:12

ArrayList數(shù)據(jù)存儲(chǔ)容器
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)