非常好,我們現在可以把我們的應用的不同部分放入不同的文件里,并且通過生成模塊的方式把它們連接到一起了。
我們仍然只擁有整個應用的最初部分:我們可以接收HTTP請求。但是我們得做點什么——對于不同的URL請求,服務器應該有不同的反應。
對于一個非常簡單的應用來說,你可以直接在回調函數 onRequest() 中做這件事情。不過就像我說過的,我們應該加入一些抽象的元素,讓我們的例子變得更有趣一點兒。
處理不同的HTTP請求在我們的代碼中是一個不同的部分,叫做“路由選擇”——那么,我們接下來就創造一個叫做 `路由` 的模塊吧。
#### 如何來進行請求的“路由”
我們要為路由提供請求的URL和其他需要的GET及POST參數,隨后路由需要根據這些數據來執行相應的代碼(這里“代碼”對應整個應用的第三部分:一系列在接收到請求時真正工作的處理程序)。
因此,我們需要查看HTTP請求,從中提取出請求的URL以及GET/POST參數。這一功能應當屬于路由還是服務器(甚至作為一個模塊自身的功能)確實值得探討,但這里暫定其為我們的HTTP服務器的功能。
我們需要的所有數據都會包含在request對象中,該對象作為`onRequest()`回調函數的第一個參數傳遞。但是為了解析這些數據,我們需要額外的Node.JS模塊,它們分別是url和querystring模塊。
~~~
url.parse(string).query
|
url.parse(string).pathname |
| |
| |
------ -------------------
http://localhost:8888/start?foo=bar&hello=world
--- -----
| |
| |
querystring(string)["foo"] |
|
querystring(string)["hello"]
~~~
當然我們也可以用querystring模塊來解析POST請求體中的參數,稍后會有演示。
現在我們來給onRequest()函數加上一些邏輯,用來找出瀏覽器請求的URL路徑:
~~~
var http = require("http");
var url = require("url");
function start() {
function onRequest(request, response) {
var pathname = url.parse(request.url).pathname;
console.log("Request for " + pathname + " received.");
response.writeHead(200, {"Content-Type": "text/plain"});
response.write("Hello World");
response.end();
}
http.createServer(onRequest).listen(8888);
console.log("Server has started.");
}
exports.start = start;
~~~
好了,我們的應用現在可以通過請求的URL路徑來區別不同請求了--這使我們得以使用路由(還未完成)來將請求以URL路徑為基準映射到處理程序上。
在我們所要構建的應用中,這意味著來自/start和/upload的請求可以使用不同的代碼來處理。稍后我們將看到這些內容是如何整合到一起的。
現在我們可以來編寫路由了,建立一個名為router.js的文件,添加以下內容:
~~~
function route(pathname) {
console.log("About to route a request for " + pathname);
}
exports.route = route;
~~~
如你所見,這段代碼什么也沒干,不過對于現在來說這是應該的。在添加更多的邏輯以前,我們先來看看如何把路由和服務器整合起來。
我們的服務器應當知道路由的存在并加以有效利用。我們當然可以通過硬編碼的方式將這一依賴項綁定到服務器上,但是其它語言的編程經驗告訴我們這會是一件非常痛苦的事,因此我們將使用依賴注入的方式較松散地添加路由模塊(你可以讀讀Martin Fowlers關于依賴注入的大作來作為背景知識)。
首先,我們來擴展一下服務器的start()函數,以便將路由函數作為參數傳遞過去:
~~~
var http = require("http");
var url = require("url");
function start(route) {
function onRequest(request, response) {
var pathname = url.parse(request.url).pathname;
console.log("Request for " + pathname + " received.");
route(pathname);
response.writeHead(200, {"Content-Type": "text/plain"});
response.write("Hello World");
response.end();
}
http.createServer(onRequest).listen(8888);
console.log("Server has started.");
}
exports.start = start;
~~~
同時,我們會相應擴展index.js,使得路由函數可以被注入到服務器中:
~~~
var server = require("./server");
var router = require("./router");
server.start(router.route);
~~~
在這里,我們傳遞的函數依舊什么也沒做。
如果現在啟動應用(node index.js,始終記得這個命令行),隨后請求一個URL,你將會看到應用輸出相應的信息,這表明我們的HTTP服務器已經在使用路由模塊了,并會將請求的路徑傳遞給路由:
~~~
bash$ node index.js
Request for /foo received.
About to route a request for /foo
~~~
(以上輸出已經去掉了比較煩人的/favicon.ico請求相關的部分)。