研究Promise 方式實(shí)現(xiàn)Node.js 實(shí)踐應(yīng)用論文
Node.js 是建立在Chrome V8 引擎的javaScript 運(yùn)行時之上的平臺, 用于構(gòu)建快速、可擴(kuò)展的Web 應(yīng)用程序. Node.js 采用單線程、事件驅(qū)動、非阻塞的I/O模型, 這些特性不僅帶來了巨大的性能提升, 還減少了多線程程序設(shè)計的復(fù)雜性, 進(jìn)而提高了開發(fā)效率,使其輕量又高效. 傳統(tǒng)的Node.js 在處理異步問題時,一般采用的是callback 回調(diào)的方式. callback 回調(diào)存在一個很嚴(yán)重的金字塔問題——大量的回調(diào)函數(shù)慢慢向右側(cè)屏幕延伸的一種狀態(tài).
Promise 是異步編程的一種解決方案, 比傳統(tǒng)的解決方案——回調(diào)函數(shù)和事件, 更合理和強(qiáng)大. 它最早由javascript 社區(qū)提出和實(shí)現(xiàn), 目前最新的JavaScript語言標(biāo)準(zhǔn)ES6 已將其寫進(jìn)了標(biāo)準(zhǔn)中, 統(tǒng)一了用法, 原生提供了Promise 對象. 借助Promise 對象, 可以將異步操作以同步操作的流程表達(dá)出來, 避免了層層嵌套的回調(diào)函數(shù).
本文就是采用Promise 方式在Node.js 平臺上搭建了一個網(wǎng)絡(luò)爬蟲的應(yīng)用. 本文首先介紹了Node.js 平臺以及其相關(guān)的一些特點(diǎn)和概念, 然后在此基礎(chǔ)上, 針對其傳統(tǒng)的callback 的回調(diào)方式的“回調(diào)地獄”等問題,引入了Promise 對象來處理這種異步回調(diào)的問題. 通過深入分析Promise 對象的理論知識以及規(guī)范, 將其合理地運(yùn)用到網(wǎng)絡(luò)爬蟲的應(yīng)用中去. 最后通過爬取一個課程網(wǎng)站的視頻課程信息, 充分展示了Node.js 平臺的強(qiáng)大和方便, 以及Promise 對象在處理異步回調(diào)問題上的優(yōu)越性以及新思路.
1.Node.js平臺介紹
Node.js 是一位叫Ryan Dahl 的程序員發(fā)明的. 他的工作是用C/C++寫高性能Web 服務(wù). 對于高性能,異步IO、事件驅(qū)動是基本原則, 但是用C/C++寫就太痛苦了. 于是Ryan 開始設(shè)想用高級語言開發(fā)Web 服務(wù).他評估了很多種高級語言, 發(fā)現(xiàn)很多語言雖然同時提供了同步IO 和異步IO, 但是開發(fā)人員一旦用了同步IO, 他們就再也懶得寫異步IO 了, 所以, 最終, Ryan瞄向了JavaScript. 因?yàn)镴avaScript 是單線程執(zhí)行, 根本不能進(jìn)行同步IO 操作, 所以, JavaScript 的這一“缺陷”導(dǎo)致了它只能使用異步IO.
選定了開發(fā)語言, 還要有運(yùn)行時引擎. Ryan 曾考慮過自己寫一個, 不過明智地放棄了, 因?yàn)閂8 就是開源的JavaScript 引擎. 讓Google 投資去優(yōu)化V8, 我們只管拿過來用就好了.于是在2009 年, Ryan 正式推出了基于JavaScript語言和V8 引擎的開源Web 服務(wù)器項(xiàng)目, 命名為Node.js. Node 第一次把JavaScript 帶入到后端服務(wù)器開發(fā), 加上世界上已經(jīng)有無數(shù)的JavaScript 開發(fā)人員,所以Node.js 一下子就火了起來.Node.js 主要特點(diǎn)是(1)時間驅(qū)動、異步編程; (2)單進(jìn)程單線程.
1.1 事件驅(qū)動、異步編程
事件驅(qū)動并不是Node.js 專屬, 在某些傳統(tǒng)語言的網(wǎng)絡(luò)編程中, 我們會用到回調(diào)函數(shù), 比如當(dāng)socket 資源達(dá)到某種狀態(tài)時, 注冊的回調(diào)函數(shù)就會執(zhí)行.Node.js 的設(shè)計思想中以事件驅(qū)動為核心, 它提供的絕大多數(shù)API 都是基于事件的、異步的風(fēng)格. 以Net 模塊為例, 其中的net.Socket 對象就有以下事件:connect、data、end、timeout、drain、error、close 等, 使用Node.js 的開發(fā)人員需要根據(jù)自己的業(yè)務(wù)邏輯注冊相應(yīng)的回調(diào)函數(shù). 這些回調(diào)函數(shù)都是異步執(zhí)行的, 這意味著雖然在代碼結(jié)構(gòu)中, 這些函數(shù)看似是依次注冊的, 但是它們并不依賴于自身出現(xiàn)的順序, 而是等待相應(yīng)的事件觸發(fā). 事件驅(qū)動、異步編程的設(shè)計重要的優(yōu)勢在于, 充分利用了系統(tǒng)資源, 執(zhí)行代碼無須阻塞等待某種操作完成, 有限的資源可以用于其他的任務(wù).此類設(shè)計非常適合于后端的網(wǎng)絡(luò)服務(wù)編程, Node.js 的目標(biāo)也在于此. 在服務(wù)器開發(fā)中, 并發(fā)的請求處理是個大問題, 阻塞式的函數(shù)會導(dǎo)致資源浪費(fèi)和時間延遲.通過事件注冊、異步函數(shù), 開發(fā)人員可以提高資源的利用率, 性能也會改善.從Node.js 提供的支持模塊中, 我們可以看到包括文件操作在內(nèi)的許多函數(shù)都是異步執(zhí)行的, 這和傳統(tǒng)語言存在區(qū)別, 而且為了方便服務(wù)器開發(fā), Node.js 的網(wǎng)絡(luò)模塊特別多, 包括HTTP、DNS、NET、UDP、HTTPS、TLS 等, 開發(fā)人員可以在此基礎(chǔ)上快速構(gòu)建Web 服務(wù)器.
1.2 單進(jìn)程單線程
1.2.1 高性能
Node.js 單線程模式避免了傳統(tǒng)php 那樣頻繁創(chuàng)建、切換線程的花銷, 執(zhí)行速度更快. 而且, 資源占用小, Node.js 在大負(fù)荷下對內(nèi)存占用任然很低.
1.2.2 線程安全
單線程的node.js 還保證了絕對的線程安全, 不用擔(dān)心統(tǒng)一變量同時被多個線程進(jìn)行讀寫而造成程序崩潰. 線程安全的同時也解放了開發(fā)人員, 免去了多線程編程中忘記對變量加鎖或者解鎖造成的隱患.
2.Promise
Promise 主要解決JavaScript 中異步的場景.Promise 是個對象, 同JavaScript 中其它對象沒什么區(qū)別, 但同時它也是一個規(guī)范, 針對異步操作約定了統(tǒng)一的接口, 表示一個一步操作最終的結(jié)果, 以同步的方式來寫代碼, 執(zhí)行的操作是異步的, 但是又保證程序的執(zhí)行順序是同步的. 這原本是JavaScript 社區(qū)的一個規(guī)范的構(gòu)想, 現(xiàn)在已經(jīng)被加入到了ES6 的語言標(biāo)準(zhǔn)中, Firefox 和Chrome 等瀏覽器已經(jīng)對它進(jìn)行了實(shí)現(xiàn).
2.1 同步與異步
JS 引擎是單線程的. 這意味著在任何環(huán)境中, 只有一段JS 代碼會被執(zhí)行. 每個函數(shù)是一個不可分割的片段或者代碼塊. 當(dāng)JS 引擎開始執(zhí)行一個函數(shù)(比如回調(diào)函數(shù))時, 它就會把這個函數(shù)執(zhí)行完, 只有執(zhí)行完這段代碼才會繼續(xù)執(zhí)行后面的代碼. 這就是JS 中的同步. Promise 對象的then()方法就是同步處理每個Promise 對象.異步是指在執(zhí)行一段代碼時, 這段代碼依賴一些其他的操作或者數(shù)據(jù), 這時就不用等待數(shù)據(jù)或者操作的返回, 直接執(zhí)行下一段代碼, 當(dāng)有數(shù)據(jù)或操作返回時再去響應(yīng)之前的代碼, 從而提高代碼執(zhí)行的效率.
2.2 Promise 對象的狀態(tài)
Promise 對象只有三種狀態(tài):
(1) Pending: 初始狀態(tài), 進(jìn)行中.
(2) Resolved(或Fulfilled): 成功的操作.
(3) Rejected: 失敗的操作.
(1) Promise 對象的狀態(tài)不受外界影響.
Promise 對象代表一個異步操作, 有三種狀態(tài):Pending(進(jìn)行中)、Resolved(已完成, 又稱Fulfilled)和Rejected(已失敗). 只有異步操作的結(jié)果, 可以決定當(dāng)前是哪一種狀態(tài), 任何其他操作都無法改變這個狀態(tài).
(2) Promise 對象一旦狀態(tài)改變, 就不會再變, 任何時候都可以得到這個結(jié)果.Promise 對象的狀態(tài)改變, 只有兩種可能: 從Pending 變?yōu)镽esolved 和從Pending 變?yōu)镽ejected. 只要這兩種情況發(fā)生, 狀態(tài)就凝固了, 不會再變了, 會一直保持這個結(jié)果. 就算改變已經(jīng)發(fā)生了, 再對Promise 對象添加回調(diào)函數(shù), 也會立即得到這個結(jié)果.
2.3 Promise 的核心方法
Promise 對象的核心部件是它的then 方法, 它的作用是為Promise 實(shí)例添加狀態(tài)改變時的回調(diào)函數(shù). then方法接受兩個回調(diào)函數(shù)作為參數(shù). 第一個回調(diào)函數(shù)是Promise 對象的狀態(tài)變?yōu)镽esolved 時調(diào)用, 第二個回調(diào)函數(shù)是Promise 對象的狀態(tài)變?yōu)镽ejected 時調(diào)用. 其中,第二個函數(shù)是可選的, 不一定要提供. 這兩個函數(shù)都接受Promise 對象傳出的值作為參數(shù).Promise 對象另一個核心方法是它的catch 方法,用于指定發(fā)生錯誤時的回調(diào)函數(shù), 是then(null,rejection)的別名. catch 方法可以捕捉promise 實(shí)例中的異常還能捕獲在它之前太狠方法中發(fā)生的異常, 所以在實(shí)際的使用中, 多用catch 方法來取代then(null,rejection)處理異常.
3.爬蟲應(yīng)用設(shè)計與實(shí)現(xiàn)
3.1 模塊加載
新建一個promise_crawler.js 文件, 首先把需要的相應(yīng)的模塊加載進(jìn)來.http 模塊: 主要用于搭建 HTTP 服務(wù)端和客戶端,使用 HTTP 服務(wù)器或客戶端功能必須調(diào)用 http 模塊;bluebird 模塊: Promise 類庫(在最新的Node.js 里已經(jīng)引入了Promise 模塊, 可直接使用, 但考慮到兼容性問題, 本例中采用bluebird 模塊);cheerio 模塊: 類似于前端的jQuery, 能夠簡單方便地操作裝在后臺的html.
3.2 組織數(shù)據(jù)結(jié)構(gòu)
首先在chrome 瀏覽器中打開需要爬取的網(wǎng)頁, 同時打開控制臺查看網(wǎng)頁html DOM 結(jié)構(gòu), 分析出所需信息, 組織好數(shù)據(jù)結(jié)構(gòu), 然后根據(jù)DOM結(jié)構(gòu)去獲取所需信息.
3.3 Promise 主要流程
本例中完成的主要功能是, 同時爬取一個課程網(wǎng)站的多個頁面, 獲取相關(guān)信息, 然后將數(shù)據(jù)按照組織好的數(shù)據(jù)結(jié)構(gòu)打印出來.代碼中所用到的Promise.all 方法用于將多個Promise 實(shí)例, 包裝成一個新的Promise 實(shí)例.該方法接收一個Promise 對象數(shù)組作為參數(shù), p1、p2、p3 都是Promise 對象的實(shí)例.p 的狀態(tài)由p1、p2、p3 決定, 分成兩種情況.
(1) 只有p1、p2、p3 的狀態(tài)都變成Resolved, p 的狀態(tài)才會變成Resolved, 此時p1、p2、p3 的返回值組成一個數(shù)組, 傳遞給p 的回調(diào)函數(shù).
(2) 只要p1、p2、p3 之中有一個被rejected, p 的狀態(tài)就變成Rejected, 此時第一個被Rejected 的實(shí)例的返回值, 會傳遞給p 的回調(diào)函數(shù).
3.4 相關(guān)函數(shù)實(shí)現(xiàn)
3.4.1 爬取頁面getPageAsync(url)
通過http 模塊的get 方法爬取頁面數(shù)據(jù), 最后返回一個Promise 對象, 方便異步處理.
3.4.2 過濾數(shù)據(jù)filterChapters(html)
過濾出每個頁面所需的`數(shù)據(jù), 然后按一定的數(shù)據(jù)結(jié)構(gòu)組織起來.
3.4.3 打印數(shù)據(jù)printCourseInfo(coursesData)
將爬取到的數(shù)據(jù), 按照組織好的數(shù)據(jù)結(jié)構(gòu)打印出來.
3.4 實(shí)驗(yàn)結(jié)果
執(zhí)行promise_crawler.js 文件, 即可看到輸出的相關(guān)信息實(shí)驗(yàn)中同爬取了4 個頁面, 可以看到, 實(shí)驗(yàn)結(jié)果是按照代碼中設(shè)定好的數(shù)據(jù)結(jié)構(gòu)爬取并打印出來的,符合實(shí)驗(yàn)預(yù)期. Promise 對象是基于異步的方式來處理程序的. 爬取每個頁面時, 不用等待頁面的數(shù)據(jù)處理完畢再去爬取下一個頁面, 而是無阻塞不間斷的去爬取每個頁面, 當(dāng)有異步的數(shù)據(jù)返回時調(diào)用Promise 對象的resolve()方法去處理, 出現(xiàn)錯誤異常時調(diào)用reject()方法去解決. 當(dāng)有多個Promise 對象時, 調(diào)用then(onFulfilled)方法, 同步處理每個Promise 對象, 一旦處理哪個Promise 對象出錯時, 可以立即調(diào)用catch方法處理異常, 中止程序往下執(zhí)行, 及時發(fā)現(xiàn)錯誤.而且onFulfilled()方法每次返回的是新的Promise 對象,這樣保證了then()可以一直鏈?zhǔn)秸{(diào)用下去, 提高了程序的效率和可靠性.
4.結(jié)語
Node.js 作為一門新興的技術(shù), 打通了前后端的界限. 由于采用事件驅(qū)動和無阻塞模型, 可以很方便的構(gòu)建高效、可擴(kuò)展的網(wǎng)絡(luò)應(yīng)用, 這是Node.js 最大的一個優(yōu)點(diǎn), 同時也是最大的一個缺點(diǎn), 由于事件驅(qū)動和無阻塞模型是建立在callback 這種回調(diào)方式上的, 隨著回調(diào)的增加, 代碼嵌套的層次就會增加, 這樣很容易陷入“回調(diào)地獄”, 這種代碼難以編寫, 難以理解而且難以維護(hù).Promise 對象是解決Node.js 中異步回調(diào)的一種很有效的方式. 借助Promise 對象, 可以將異步操作以同步操作的流程表達(dá)出來, 避免了層層嵌套的回調(diào)函數(shù).在保證異步回調(diào)的基礎(chǔ)上又實(shí)現(xiàn)了多個promise 對象之間的同步順序, 使程序能快速高效的執(zhí)行下去, 給我們的開發(fā)帶來很大的便利.
【研究Promise 方式實(shí)現(xiàn)Node.js 實(shí)踐應(yīng)用論文】相關(guān)文章:
培養(yǎng)高職學(xué)生主動意識的實(shí)踐與探索的方式研究論文12-11
自我傳播的途徑與方式研究論文11-03
轉(zhuǎn)變農(nóng)業(yè)發(fā)展方式研究論文05-23
中職學(xué)生行為方式研究論文06-18
關(guān)于PLC實(shí)現(xiàn)高爐料流閥自修正控制的研究與應(yīng)用的論文07-18
預(yù)防醫(yī)學(xué)實(shí)踐教學(xué)改革的應(yīng)用研究論文07-31
Proteus在電子實(shí)踐教學(xué)課程中應(yīng)用的研究論文01-23