[技術翻譯]在現代JavaScript中編寫異步任務

本週再來翻譯一些技術文章,本次預計翻譯三篇文章如下:

我翻譯的技術文章都放在一個github倉庫中,如果覺得有用請點擊star收藏。我爲什麼要創建這個git倉庫?目的是通過翻譯國外的web相關的技術文章來學習和跟進web發展的新思想和新技術。git倉庫地址:https://github.com/yzsunlei/javascript-article-translate

在本文中,我們將探討過去圍繞異步執行的JavaScript的演變以及它如何改變我們編寫和讀取代碼的方式。我們將從Web開發的開始,一直到現代異步模式示例。
JavaScript作爲編程語言具有兩個主要特徵,這兩個特徵對於理解我們的代碼是如何工作的都很重要。首先是它的同步特性,這意味着代碼將幾乎在您閱讀時逐行運行,其次,它是單線程的,任何時候都只執行一個命令。

隨着語言的發展,新的模塊出現在場景中以允許異步執行。開發人員在解決更復雜的算法和數據流時嘗試了不同的方法,從而導致圍繞它們的新接口和模式的出現。

同步執行和觀察者模式

如引言中所述,JavaScript通常會逐行運行您編寫的代碼。即使在最初的幾年中,該語言也有例外,儘管它們很少,您可能已經知道它們:HTTP請求,DOM事件和時間間隔。

const button = document.querySelector('button');

// observe for user interaction
button.addEventListener('click', function(e) {
  console.log('user click just happened!');
})

如果添加事件偵聽器(例如,單擊元素並觸發用戶交互),則JavaScript引擎會將事件偵聽器回調的任務放入隊列,但將繼續執行其當前堆棧中的內容。完成那裏的調用之後,它現在將運行偵聽器的回調。

此行爲類似於網絡請求和計時器發生的情況,它們是Web開發人員訪問異步執行的第一個模塊。

儘管這些是JavaScript中常見的同步執行例外的,但至關重要的是要了解該語言仍然是單線程的,並且儘管它可以將Task排隊,異步運行它們然後返回主線程,但它只能一次執行一段代碼。

我們的工具手冊,其中Alla Kholmatova探索瞭如何創建有效且可維護的設計系統來設計出色的數字產品。認識Design Systems,瞭解常見的陷阱,陷阱和Alla多年來汲取的經驗教訓。

例如,讓我們發送一個網絡請求。

var request = new XMLHttpRequest();
request.open('GET', '//some.api.at/server', true);

// observe for server response
request.onreadystatechange = function() {
  if (request.readyState === 4 && xhr.status === 200) {
    console.log(request.responseText);
  }
}

request.send();

服務器返回時,分配給該方法的任務將onreadystatechange放入隊列(代碼在主線程中繼續執行)。

注意:解釋JavaScript引擎如何將任務排隊和處理執行線程是一個很複雜的主題,可能值得一讀。不過,我還是建議您查看“事件循環到底是什麼?”菲利普·羅伯茨(Phillip Roberts)提供的幫助,以幫助您更好地理解。

在上述每種情況下,我們都在響應外部事件。達到一定的時間間隔,用戶操作或服務器響應。我們本身無法創建異步任務,我們始終觀察到發生的事件超出了我們的範圍。

這就是爲什麼將這種模板式的代碼稱爲“觀察者模式”,addEventListener在這種情況下,可以更好地由接口表示。很快,暴露這種模式的事件庫或框架蓬勃發展。

NODE.JS和事件觸發器

一個很好的例子是Node.js,該頁面將自己描述爲“異步事件驅動的JavaScript運行時”,因此事件觸發器和回調是一等公民。它甚至用EventEmitter已經實現了一個構造函數。

const EventEmitter = require('events');
const emitter = new EventEmitter();

// respond to events
emitter.on('greeting', (message) => console.log(message));

// send events
emitter.emit('greeting', 'Hi there!');

這不僅是異步執行的通用方法,而且是其生態系統的核心模式和慣例。Node.js開闢了一個在不同環境中甚至在網絡之外編寫JavaScript的新時代。結果,其他異步情況也是可能的,例如創建新目錄或寫入文件。

const { mkdir, writeFile } = require('fs');

const styles = 'body { background: #ffdead; }';

mkdir('./assets/', (error) => {
  if (!error) {
    writeFile('assets/main.css', styles, 'utf-8', (error) => {
      if (!error) console.log('stylesheet created');
    })
  }
})

您可能會注意到,回調error函數的第一個參數爲,如果需要響應數據,則將其作爲第二個參數。這被稱爲“錯誤優先回調模式”,它成爲作者和貢獻者爲其自己的程序包和庫所採用的約定。

Promise和無盡的回調鏈

隨着Web開發面臨更復雜的問題需要解決,對更好的異步工件的需求出現了。如果我們查看最後一個代碼片段,我們會看到重複的回調鏈,隨着任務數量的增加,回調鏈的擴展效果就很差。

例如,讓我們僅添加兩個步驟,即文件讀取和樣式預處理。

const { mkdir, writeFile, readFile } = require('fs');
const less = require('less')

readFile('./main.less', 'utf-8', (error, data) => {
  if (error) throw error
  less.render(data, (lessError, output) => {
    if (lessError) throw lessError
    mkdir('./assets/', (dirError) => {
      if (dirError) throw dirError
      writeFile('assets/main.css', output.css, 'utf-8', (writeError) => {
        if (writeError) throw writeError
        console.log('stylesheet created');
      })
    })
  })
})

我們可以看到,由於多個回調鏈和重複的錯誤處理,隨着正在編寫的程序變得越來越複雜,代碼變得更加難以爲人所知。

Promise,包裝和連鎖模式

Promises最初宣佈它們是JavaScript語言的新功能時,並沒有引起太多關注,它們並不是一個新概念,因爲其他語言在幾十年前就已經實現了類似的功能。事實是,自從出現以來,他們發現我所做的大多數項目的語義和結構都發生了很大變化。

Promises不僅引入了供開發人員編寫異步代碼的內置解決方案,而且還爲Web開發(如Web規範)的新功能的構建基礎打開了Web開發的新階段fetch。

從回調方法遷移到基於Promise的方法在項目(例如庫和瀏覽器)中變得越來越普遍,甚至Node.js也開始緩慢地遷移到它們。

例如,包裝一下Node的readFile方法:

const { readFile } = require('fs');

const asyncReadFile = (path, options) => {
  return new Promise((resolve, reject) => {
    readFile(path, options, (error, data) => {
      if (error) reject(error);
      else resolve(data);
    })
  });
}

在這裏,我們通過在Promise構造函數中執行,resolve在方法結果成功時以及reject在定義錯誤對象時調用,來掩蓋回調。

當一個方法返回一個Promise對象時,我們可以通過將一個函數傳遞給來遵循其成功的解析then,其參數是Promise被解析的值,在這種情況下爲data。

如果在方法期間引發錯誤catch,則將調用該函數(如果存在)。

注意:如果您需要更深入地瞭解Promises的工作方式,我建議Jake Archibald 在Google的Web開發博客上寫的“JavaScript Promises:Introduction”一文。

現在我們可以使用這些新方法並避免回調鏈。

asyncRead('./main.less', 'utf-8')
  .then(data => console.log('file content', data))
  .catch(error => console.error('something went wrong', error))

具有創建異步任務的方法和清晰的界面以跟蹤其可能的結果,使該行業擺脫了觀察者模式。基於Promise的代碼似乎可以解決不可讀且容易出錯的代碼。

隨着更好的語法或更清晰的錯誤消息在編碼時突出顯示有所幫助,對於開發人員來說,更易於推理的代碼變得更具可預測性,並且執行路徑的情況更好,更容易捕捉可能的代碼陷阱。

Promises由於在社區中的普及程度很高,Node.js迅速發佈了其I/O方法的內置版本以返回Promise對象,例如從中導入文件操作fs.promises。

它甚至提供了一個promisify實用工具,用於包裝遵循錯誤優先回調模式的所有函數,並將其轉換爲基於Promise的函數。

但是,Promises在所有情況下都能提供幫助嗎?

讓我們重新想象一下用Promises編寫的樣式預處理任務。

const { mkdir, writeFile, readFile } = require('fs').promises;
const less = require('less')

readFile('./main.less', 'utf-8')
  .then(less.render)
  .then(result =>
    mkdir('./assets')
      .then(writeFile('assets/main.css', result.css, 'utf-8'))
  )
  .catch(error => console.error(error))

代碼中的冗餘明顯減少了,尤其是在我們現在所依賴的錯誤處理方面catch,但是Promises某種程度上未能提供與操作串聯直接相關的清晰代碼縮進。

這實際上是在調用then之後的第一個語句上實現的readFile。這些行之後發生的事情是需要創建一個新的作用域,我們可以在該作用域中首先創建目錄,然後將結果寫入文件中。這就導致了縮進節奏的中斷,乍看之下很難確定指令序列。

解決此問題的一種方法是預先處理該問題的自定義方法,並允許該方法正確連接,但是我們將向似乎已經具有實現任務所需功能的代碼引入更多的複雜性。

注意:這是一個示例程序,我們可以控制某些方法,它們都遵循行業慣例,但並非總是如此。通過更復雜的串聯或引入具有不同類型的庫,我們可以輕鬆破壞代碼風格。

令人高興的是,JavaScript社區再次從其他語言語法中學到了東西,並添加了一種表示法,可以在很多情況下幫助異步任務串聯而不是像同步代碼那樣令人愉悅或直截了當。

async和await

A Promise在執行時被定義爲一個未解析的值,創建a的實例Promise是對該模塊的顯式調用。

const { mkdir, writeFile, readFile } = require('fs').promises;
const less = require('less')

readFile('./main.less', 'utf-8')
  .then(less.render)
  .then(result =>
    mkdir('./assets')
      .then(writeFile('assets/main.css', result.css, 'utf-8'))
  )
  .catch(error => console.error(error))

在async方法內部,我們可以使用await保留字來確定a的分辨率,Promise然後繼續執行。

讓我們使用此語法重新訪問或編寫代碼段。

const { mkdir, writeFile, readFile } = require('fs').promises;
const less = require('less')

async function processLess() {
  const content = await readFile('./main.less', 'utf-8')
  const result = await less.render(content)
  await mkdir('./assets')
  await writeFile('assets/main.css', result.css, 'utf-8')
}

processLess()

注意:請注意,由於我們今天不能在異步函數的範圍之外使用,因此需要將所有代碼移至方法await。

每次async方法找到一條await語句時,它將停止執行,直到處理中的值或Promise被解析爲止。

儘管異步執行,但使用async/await表示法會有明顯的後果,代碼看起來好像是async,這是我們開發人員更習慣查看和推理的。

錯誤處理呢?爲此,我們使用在該語言中已經存在很長時間的語句,try和catch。

const { mkdir, writeFile, readFile } = require('fs').promises;
const less = require('less');

async function processLess() {
  try {
    const content = await readFile('./main.less', 'utf-8')
    const result = await less.render(content)
    await mkdir('./assets')
    await writeFile('assets/main.css', result.css, 'utf-8')
  } catch(e) {
    console.error(e)
  }
}

processLess()

放心,在該過程中引發的任何錯誤將由該catch語句內的代碼處理。我們在中心位置負責錯誤處理,但是現在我們有了一個易於閱讀和遵循的代碼。

具有返回值的後續操作不需要存儲在mkdir不會破壞代碼節奏的變量中;也無需在以後的步驟中創建新的作用域來訪問result的值。

可以肯定地說,Promises是該語言中引入的一個基本模塊,對於在JavaScript中啓用async/await表示法是必需的,您可以在現代瀏覽器和最新版本的Node.js中使用它。

注意:最近在JSConf中,Node的創建者和第一貢獻者Ryan Dahl很遺憾沒有堅持Promises的早期開發,主要是因爲Node的目標是創建事件驅動的服務器和文件管理,而Observer模式更適合於此。

結論

將Promises引入Web開發世界的目的是改變我們在代碼中排隊操作的方式,並改變了我們對代碼執行進行推理的方式以及我們編寫庫和包的方式。

但是擺脫回調鏈很難解決,我認爲then在多年習慣於觀察者模式和主要提供商採用的方法之後,不得不通過一種方法並不能幫助我們擺脫思路。像Node.js這樣的社區。

正如諾蘭·勞森(Nolan Lawson)在其有關Promise串聯中錯誤使用的出色文章中所說,舊的回調習慣會死掉!稍後,他解釋瞭如何避免這些陷阱。

我認爲Promises是中間步驟,它允許以自然的方式生成異步任務,但並沒有幫助我們進一步改進更好的代碼模式,有時您實際上需要更適應和改進的語言語法。

當我們嘗試使用JavaScript解決更復雜的難題時,我們看到了對更成熟語言的需求,並嘗試了以前不曾在網絡上看到過的架構和模式。

我們仍然不知道ECMAScript規範的表現如何,因爲我們一直將JavaScript治理擴展到網絡之外,並嘗試解決更復雜的難題。

現在很難說我們需要從語言中真正地將這些難題轉變成更簡單的程序所需要的東西,但是我對Web和JavaScript本身如何推動事物,試圖適應挑戰和新環境感到滿意。與十年前開始在瀏覽器中編寫代碼相比,現在我覺得JavaScript是一個更加異步的友好的地方。

原文鏈接:https://www.smashingmagazine.com/2019/10/asynchronous-tasks-modern-javascript/

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