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

基于Node.js、Express和Jscex開(kāi)發(fā)的ToDo網(wǎng)站示例

開(kāi)發(fā) 前端
目前用Node.js開(kāi)發(fā)網(wǎng)站最著名的框架是Express,使用起來(lái)也是比較容易的。前段時(shí)間看到CNodeJS社區(qū)的一篇文章,有同學(xué)將一個(gè)Python寫的ToDo列表網(wǎng)站移植到了Node.js上,我為了推廣Jscex,就fork了這個(gè)項(xiàng)目,將其修改為基于Jscex的版本,大伙兒可以來(lái)比較一下。

Jscex的主要使用場(chǎng)景是“JavaScript異步編程”,不過(guò)并沒(méi)有限制是跑在瀏覽器還是服務(wù)器端。最近Node.js很火熱,也剛發(fā)布了原生的Windows版,不少同學(xué)會(huì)用它來(lái)做一些網(wǎng)站這樣的小程序。目前用Node.js開(kāi)發(fā)網(wǎng)站***的框架是Express,使用起來(lái)也是比較容易的。前段時(shí)間看到CNodeJS社區(qū)的一篇文章,有同學(xué)將一個(gè)Python寫的ToDo列表網(wǎng)站移植到了Node.js上,我為了推廣Jscex,就fork了這個(gè)項(xiàng)目,將其修改為基于Jscex的版本,大伙兒可以來(lái)比較一下。當(dāng)然這個(gè)網(wǎng)站過(guò)于簡(jiǎn)單,我也正在尋找更合適的項(xiàng)目。

51CTO推薦專題:Node.js專區(qū)

JavaScript是一個(gè)沒(méi)有阻塞特性的語(yǔ)言,因此各類API都會(huì)設(shè)計(jì)為異步,這對(duì)于服務(wù)器的伸縮性和客戶端網(wǎng)頁(yè)的響應(yīng)能力都有好處,不過(guò)在程序編寫上就會(huì)遇到各種問(wèn)題了。例如在ToDo示例中的一個(gè)簡(jiǎn)單的處理函數(shù),因?yàn)樾枰樵償?shù)據(jù)庫(kù),就要寫成帶回調(diào)的樣子:

  1. exports.index = function (req, res, next) {  
  2.     db.query('select * from todo order by finished asc, id asc limit 50'function (err, rows) {  
  3.         if (err) return next(err);  
  4.         res.render('index', { todos: rows });  
  5.     });  
  6. }; 

db變量用來(lái)操作MySQL數(shù)據(jù)庫(kù),它的query方法傳入sql(可能還會(huì)有參數(shù))并提供一個(gè)回調(diào)函數(shù),用來(lái)提示錯(cuò)誤或是返回查詢結(jié)果。在回調(diào)中我們必須判斷err是否存在,如果存在便調(diào)用next報(bào)告框架“出錯(cuò)了”。每個(gè)異步操作都必須如此,試想如果在這個(gè)查詢后還有另一個(gè)查詢,則還需要進(jìn)行一次嵌套和err判斷。每個(gè)處理函數(shù)都是如此,這也是異步編程的煩惱之一:難以進(jìn)行統(tǒng)一的異常處理,處理代碼總是需要分散在各處,一不小心就變成“野異常”,還很難排查出來(lái)。

我將ToDo網(wǎng)站簡(jiǎn)單地Jscex化了一下。首先讓MySQL的查詢能夠接入Jscex(lib\jscex.mysql.js):

  1. exports.jscexify = function (db) {  
  2.     db.queryAsync = function () {  
  3.         var _this = this;  
  4.  
  5.         var args = [];  
  6.         for (var i = 0; i < arguments.length; i++) {  
  7.             args.push(arguments[i]);  
  8.         }  
  9.  
  10.         var delegate = {  
  11.             onStart: function (callback) {  
  12.  
  13.                 args.push(function (err, result) {  
  14.                     if (err) {  
  15.                         callback("failure", err);  
  16.                     } else {  
  17.                         callback("success", result);  
  18.                     }  
  19.                 });  
  20.  
  21.                 _this.query.apply(_this, args);  
  22.             }  
  23.         };  
  24.  
  25.         return new Jscex.Async.Task(delegate);  
  26.     }  

一般來(lái)說(shuō),將一個(gè)異步接口給Jscex化并不需要那么多代碼(最關(guān)鍵的其實(shí)只是onStart函數(shù))。這里近30行代碼,其中大部分是為了支持“變長(zhǎng)”參數(shù),因此queryAsync函數(shù)會(huì)保留調(diào)用時(shí)的所有參數(shù),補(bǔ)上一個(gè)callback,再去調(diào)用query函數(shù)本身。此時(shí),便可以去改寫之前的index等處理函數(shù)了(controllers\todo.js),例如:

  1. exports.index = toHandler(eval(Jscex.compile("async"function (req, res) {  
  2.  
  3.     var todos = $await(db.queryAsync('select * from todo order by finished asc, id asc limit 50'));  
  4.     res.render("index", { todos: todos });  
  5.  
  6. }))); 

toHandler函數(shù)的作用,是將一個(gè)“接受req和res,返回Task”的函數(shù),封裝成標(biāo)準(zhǔn)的“接受req、res和next三個(gè)參數(shù)”的處理函數(shù),并提供統(tǒng)一的錯(cuò)誤處理:

  1. var toHandler = function (asyncFunc) {  
  2.     return function (req, res, next) {  
  3.         var task = asyncFunc(req, res);  
  4.         task.addListener(function () {  
  5.             if (task.status == "failed") {  
  6.                 next(task.error);  
  7.             }  
  8.         });  
  9.         task.start();  
  10.     }  

我在todo.js里保留了原有各個(gè)處理函數(shù)的實(shí)現(xiàn),感興趣的朋友可以對(duì)比一下它們之前的差別。可惜的是,由于ToDo實(shí)在過(guò)于簡(jiǎn)單,Jscex的優(yōu)勢(shì)并沒(méi)有表現(xiàn)出來(lái)太多。例如,每個(gè)處理程序中只有一個(gè)MySQL查詢,沒(méi)有判斷和循環(huán),更別說(shuō)為了充分利用IO并發(fā)能力,從而組合多個(gè)異步函數(shù)了。因此,我最近也一直在尋找更復(fù)雜一些的示例,不過(guò)似乎用Express的開(kāi)源網(wǎng)站并不多見(jiàn),我?guī)缀醵枷胱约簩懸粋€(gè)了。目前感覺(jué)Nodepad似乎還算不錯(cuò),接下來(lái)可能會(huì)對(duì)它下手。

ToDo網(wǎng)站依賴Express,ejs和MySQL驅(qū)動(dòng),同時(shí)我把Jscex作為添加為它的子模塊。如果您要克隆一份ToDo的代碼把玩一番,可以:

  1. > git clone git://github.com/JeffreyZhao/todo.git  
  2. > cd todo  
  3. > git submodule init  
  4. > git submodule update  
  5. > npm install express ejs mysql  
  6. > node server.js 

從現(xiàn)在開(kāi)始,我會(huì)在InfoQ中文站上發(fā)表一系列關(guān)于Jscex的文章,既有關(guān)于瀏覽器端的JavaScript開(kāi)發(fā),也有在服務(wù)器端利用Node.js開(kāi)發(fā)的內(nèi)容??赡苣壳斑€可能會(huì)有所疑惑,例如為什么要使用危險(xiǎn)的eval函數(shù),eval和Jscex.compile函數(shù)不能封裝起來(lái)嗎?其實(shí)在看了我的文章并對(duì)Jscex有了基本了解之后,就會(huì)發(fā)現(xiàn)這些都是以“傳統(tǒng)眼光”來(lái)看待Jscex時(shí)所形成的誤解。Jscex的做法的確“另辟蹊徑”,否則在JavaScript異步類庫(kù)已經(jīng)多如牛毛的情況下,我不知如何讓它脫穎而出。

原文:http://blog.zhaojie.me/2011/07/nodejs-express-jscex-demo-website-todo.html

【編輯推薦】

  1. 使用Jscex改善JavaScript異步編程體驗(yàn)
  2. 揭秘Node.js事件
  3. Node.js入門之神秘的服務(wù)器端JavaScript
  4. 什么是Node.js?
  5. 親愛(ài)的PHP我要離開(kāi)你 因?yàn)槲覍?duì)NodeJs更有感
責(zé)任編輯:陳貽新 來(lái)源: 老趙的博客
相關(guān)推薦

2014-07-11 14:16:15

AbsurdJSExpress

2015-03-10 10:59:18

Node.js開(kāi)發(fā)指南基礎(chǔ)介紹

2021-12-25 22:29:57

Node.js 微任務(wù)處理事件循環(huán)

2021-08-20 16:05:28

JavaScript node.js 應(yīng)用安全

2020-08-07 10:40:56

Node.jsexpress前端

2017-04-24 08:31:26

Node.jsExpress.jsHTTP

2013-11-01 09:34:56

Node.js技術(shù)

2019-07-09 14:50:15

Node.js前端工具

2022-01-07 08:00:00

Node.js開(kāi)發(fā)Web

2011-10-25 09:28:30

Node.js

2011-11-10 11:08:34

Node.js

2021-01-14 10:48:34

Docker CompNode.js開(kāi)發(fā)

2014-09-01 10:22:11

Node.js技術(shù)架構(gòu)

2021-12-28 20:04:23

Node.js開(kāi)發(fā)JavaScript

2020-05-29 15:33:28

Node.js框架JavaScript

2012-09-29 11:13:15

Node.JS前端開(kāi)發(fā)Node.js打包

2012-02-03 09:25:39

Node.js

2014-02-19 16:28:53

Node.jsWeb工具

2024-09-25 08:04:58

2017-04-20 12:30:57

聲明式爬蟲(chóng)網(wǎng)絡(luò)
點(diǎn)贊
收藏

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