讓 Generator 自啟動

来源:http://www.cnblogs.com/52cik/archive/2016/07/11/generator-co.html
-Advertisement-
Play Games

文章同步自個人博客:http://www.52cik.com/2016/07/11/generator co.html 此前只是簡單使用而沒有真正的去研究 Generator,這次要好好折騰下這貨。 <! more 非同步編程 對於 jser 來說,非同步非常熟悉了吧,但是真正理解非同步的卻不多,因為大部 ...


文章同步自個人博客:http://www.52cik.com/2016/07/11/generator-co.html

此前只是簡單使用而沒有真正的去研究 Generator,這次要好好折騰下這貨。

非同步編程

對於 jser 來說,非同步非常熟悉了吧,但是真正理解非同步的卻不多,因為大部分人只知道回調。
隨著js的快速發展,非同步方案也層出不窮,從最開始的回調到Promise,再到Generator,然後到async/await。
甚至有人說 async/await 是非同步的終極解決方案,我不敢直接贊同,只能說是目前最好的非同步體驗。
本篇先從 Generator 講起,後序再詳細說 async/await。

從回調開始

從最最經典的 ajax 請求開始今天的話題吧。
假如,我們要依次請求 url1, url2, url3 這3個地址。

$.get('url1', function(r1) {
  $.get('url2', function(r2) {
    $.get('url3', function(r3) {
      console.log(r1, r2, r3);
    });
  });
});

一不小心就寫成這樣了。
如果你是 jQuery 粉的話,你可能會說也可以這樣實現啊。

$.get('url1').then(function(r1) {
  console.log(r1);
  return $.get('url2');
}).then(function(r2) {
  console.log(r2);
  return $.get('url3');
}).then(function(r3) {
  console.log(r3);
});

用 jQuery 的 Deferred 對象,類似 Promise 來規避回調地獄,看著確實平了,但體驗並不是特別友好。

用 Generator 來和諧回調

Generator 的基礎這裡就不展開說了,直接說應用。

function* gen() {
  var r1 = yield $.get('url1');
  var r2 = yield $.get('url2');
  var r3 = yield $.get('url3');

  console.log(r1, r2, r3);
}

這是比較友好的非同步方式,但是還有個至關重要的因素,怎麼運行這個 Generator 是個問題。
直接手動 g.next() 運行那肯定不行,鬼知道有多少個 yield。
我們要實現一個啟動器來運行它,並把 Promise 結果傳給下一次next,這樣就實現了 yield 接收值的功能。

先來實現一個最簡陋的起動器。

function run(gen) {
  var g = gen();

  function next(d) {
    var r = g.next(d);
    r.done || r.value.then(function(d){ next(d) }); // 這個是關鍵,把值傳回傳
  }

  next();
}

然後我們只要一行代碼。

run(gen);

Generator 就啟動起來了,並且一直執行到 done 為 true 為止。

真實例子

打開 http://www.52tian.net/ 動漫網。非廣告,確實沒找到合適的測試站,湊合下吧。
然後把下麵代碼貼到控制台,看下結果。如果執行不了,請升級瀏覽器,本例在 chrome 51 下通過。

function* gen() {
  var r1 = yield $.get('/json/anime/4126.htm');
  var r2 = yield $.get('/json/anime/11129.htm');
  var r3 = yield $.get('/json/anime/427.htm');

  console.log([r1, r2, r3].join('\n'));
}

function run(gen) {
  var g = gen();

  function next(d) {
    var r = g.next(d);
    r.done || r.value.then(function(d){ next(d) }); // 這個是關鍵,把值傳回傳
  }

  next();
}

run(gen);

小結

可能你已經發現了,其實這就是 co 的原理,但 co 比這個例子嚴謹多了,而且api設計的也非常友好。
本篇到此也就結束了,利用 Generator 的 yield 功能實現參數回傳,讓代碼看起來非常‘同步’,讓非同步體驗變的更加友好。


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • ...
  • 認識觀察者模式 首先來看看報紙訂閱的過程 1.報社的業務就是出版報紙 2.向某家報社訂閱報紙,只要他們有新報紙出版,就會送過來,只要你是他們的訂戶 3.當你不想再看報紙的時候,取消訂閱,他們就不會再送新報紙 4.只要報社還在運營,就會一直有人來訂閱或取消訂閱報紙 觀察者模式和報紙訂閱流程是一樣的,只 ...
  • 在上一篇《html+ccs3太陽系行星運轉動畫》中實現了太陽系八大行星的基本運轉動畫。 太陽系又何止這些內容,為豐富一下動畫,接下來增加“土星環”和“月球”來充盈太陽系動畫。 下麵是充盈後的動畫效果靜態圖。 一、土星環 修改原來土星的div,在外面放一個包裹層div,class設成saturn-co ...
  • Functionde 對象的實例可以創建構造函數 但是Object對象就不能,當完成一個object對象的實例化後,不能再基於新實例使用new 創建一個實例 添加公有方法 要在構造函數的新實例中添加公有方法,使用點號想它的原型屬性添加即可。 但是不能直接在 Myfunc 上運用 如:Myfunc.n ...
  • ...
  • 含義:滾動條高度 作用:滾動載入(ajax),滾動導航固定定位,滾動彈框定位等等. 展示滾動導航和側邊欄滾動固定定位的效果:查看效果 1、chrome瀏覽器 2、各瀏覽器下 scrollTop的差異 IE6/7/8/9/10: 對於沒有doctype聲明的頁面里可以使用 document.body. ...
  • 關於前端對話框、消息框的優秀插件多不勝數。造輪子是為了更好的使用輪子,並不是說自己造的輪子肯定好。所以,這個博客系統基本上都是自己實現的,包括日誌記錄、響應式佈局等等一些本可以使用插件的。好了,廢話不多時。我們來實現自己的對話框和消息框。 ...
  • jQuery 的 setter/getter 共用一個函數,通過是否傳參來表明它是何種意義。簡單說傳參它是 setter,不傳它是 getter。 一個函數具有多種意義在編程語言中並不罕見,比如函數重載:一組具有相同函數名,不同參數列表的函數,這組函數被稱為重載函數。重載的好處是減少了函數名的數量, ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...