默認情況下,express的路由寫起來還挺麻煩的。比如下面這樣:
app.get('/blacklists/', get_all);
app.get('/blacklists/:id', get_all);
app.post('/blacklists/:id', update);
app.post('/blacklists', create);
app.detete('/blacklists/:id' : del);
這樣寫是什么大的問題,至少它能正常運行。但有二個小問題:
1、不便於以后的擴展和維護,如果需要監聽的URL越來越多,它看上去就會越來越糟糕了;
2、不夠靈活,如果還需要對“hello”、“aa”…進行監聽,所有的代碼處理都被放在一個地方了,只會越來越臃腫;
有沒有比較好的方案對路由的處理管理和配置呢?
我們可以使用“慣例優先原則”來定義好一個約定:我期望的是所有的路由相關的處理,都放在項目文件路徑下,一個名為“routes”的文件夾里,里面可以可以再建立N層文件夾,而每一個js文件,僅處理以該文件名為路徑的請求,如上面的“blacklists”相關的處理全部放在blacklists.js文件內。如何實現呢?
1、獲取當前項目routes目錄內所有的文件(包含子文件夾);
2、加載每一個文件,獲取指定屬性下的配置,如果存在,就動態拼接成如文章開頭的配置,如:app[method](path, func);
部分實現代碼:
//假設文件存放在routes目錄,取名為hello.js
exports.autoroute = {
'get' : {
'/hello(/?)' : hello,
'/hello/:id' : hello2
},
'post' : {
}
}
function hello(req, res) {
res.end('hello');
}
function hello2(req, res) {
res.end('hello ' + req.params.id);
}
------------------------------------------------------------------------
//當前項目的routes目錄的路徑
var routesDir = path.dirname(require.main.filename) + "/routes";
fs.readdir(routesDir, function(err, files) {
if (err) {
return ;
}
files.forEach(function(path) {
//routes目錄下的文件路徑
var filePath = routesDir + "/" + path;
fs.stat(filePath, function(err, stats) {
if (err) {
return ;
}
if (stats.isDirectory()) {
//遞歸執行函數
} else {
//加載文件並解析
loadFile(filePath);
}
})
});
})
function loadFile(filePath) {
var routeObj = require(filePath);
//如果包含autoroute屬性,則進行解析
if (routeObj.autoroute) {
/*
* autoroute就是上面hello.js的內容:
'get' : {
'/hello(/?)' : hello,
'/hello/:id' : hello2
},
'post' : {
}
*/
for (var method in routeObj.autoroute) {
var routeList = routeObj.autoroute[method];
if (!routeList) {
break ;
}
//method就是上面取到的get、post
for(var routeRule in routeList) {
//func獲取得到的就是上面對應各項的處理函數
var func = routeList[path];
app[method](routeRule, func);
}
}
}
}
原文地址:http://www.cnblogs.com/meteoric_cry/archive/2013/02/19/2917639.html
