使用Node.js原生API寫一個web伺服器
- 2020 年 10 月 26 日
- 筆記
- javascript, Node.js, 前端
Node.js
是JavaScript
基礎上發展起來的語言,所以前端開發者應該天生就會一點。一般我們會用它來做CLI工具
或者Web伺服器
,做Web伺服器
也有很多成熟的框架,比如Express
和Koa
。但是Express
和Koa
都是對Node.js
原生API
的封裝,所以其實不藉助任何框架,只用原生API
我們也能寫一個Web伺服器
出來。本文要講的就是不藉助框架,只用原生API
怎麼寫一個Web伺服器
。因為在我的計劃中,後面會寫Express
和Koa
的源碼解析,他們都是使用原生API來實現的。所以本文其實是這兩個源碼解析的前置知識,可以幫我們更好的理解Express
和Koa
這種框架的意義和源碼。本文僅為說明原生API的使用方法,程式碼較丑,請不要在實際工作中模仿!
本文可運行程式碼示例已經上傳GitHub,大家可以拿下來玩玩://github.com/dennis-jiang/Front-End-Knowledges/tree/master/Examples/Node.js/HttpServer
Hello World
要搭建一個簡單的Web伺服器
,使用原生的http
模組就夠了,一個簡單的Hello World
程式幾行程式碼就夠了:
const http = require('http')
const port = 3000
const server = http.createServer((req, res) => {
res.statusCode = 200
res.setHeader('Content-Type', 'text/plain')
res.end('Hello World')
})
server.listen(port, () => {
console.log(`Server is running on //127.0.0.1:${port}/`)
})
這個例子就很簡單,直接用http.createServer
創建了一個伺服器,這個伺服器也沒啥邏輯,只是在訪問的時候返回Hello World
。伺服器創建後,使用server.listen
運行在3000
埠就行。
這個例子確實簡單,但是他貌似除了輸出一個Hello World
之外,啥也幹不了,離我們一般使用的Web伺服器
還差了很遠,主要是差了這幾塊:
- 不支援
HTTP
動詞,比如GET
,POST
等- 不支援路由
- 沒有靜態資源託管
- 不能持久化數據
前面三點是一個Web伺服器
必備的基礎功能,第四點是否需要要看情況,畢竟目前很多Node
的Web伺服器
只是作為一個中間層,真正跟資料庫打交道做持久化的還是各種微服務,但是我們也應該知道持久化怎麼做。
所以下面我們來寫一個真正能用的Web伺服器
,也就是說把前面缺的幾點都補上。
處理路由和HTTP動詞
前面我們的那個Hello World
也不是完全不能用,因為程式碼位置還是得在http.createServer
裡面,我們就在裡面添加路由的功能。為了跟後面的靜態資源做區分,我們的API請求都以/api
開頭。要做路由匹配也不難,最簡單的就是直接用if
條件判斷就行。為了能拿到請求地址,我們需要使用url
模組來解析傳過來的地址。而Http
動詞直接可以用req.method
拿到。所以http.createServer
改造如下:
const url = require('url');
const server = http.createServer((req, res) => {
// 獲取url的各個部分
// url.parse可以將req.url解析成一個對象
// 裡面包含有pathname和querystring等
const urlObject = url.parse(req.url);
const { pathname } = urlObject;
// api開頭的是API請求
if (pathname.startsWith('/api')) {
// 再判斷路由
if (pathname === '/api/users') {
// 獲取HTTP動詞
const method = req.method;
if (method === 'GET') {
// 寫一個假數據
const resData = [
{
id: 1,
name: '小明',
age: 18
},
{
id: 2,
name: '小紅',
age: 19
}
];
res.setHeader('Content-Type', 'application/json')
res.end(JSON.stringify(resData));
return;
}
}
}
});
現在我們訪問/api/users
就可以拿到用戶列表了:
支援靜態文件
上面說了API
請求是以/api
開頭,也就是說不是以這個開頭的可以認為都是靜態文件,不同文件有不同的Content-Type
,我們這個例子裡面暫時只支援一種.jpg
吧。其實就是給我們的if (pathname.startsWith('/api'))
加一個else
就行。返回靜態文件需要:
- 使用
fs
模組讀取文件。- 返迴文件的時候根據不同的文件類型設置不同的
Content-Type
。
所以我們這個else
就長這個樣子:
// ... 省略前後程式碼 ...
else {
// 使用path模組獲取文件後綴名
const extName = path.extname(pathname);
if (extName === '.jpg') {
// 使用fs模組讀取文件
fs.readFile(pathname, (err, data) => {
res.setHeader('Content-Type', 'image/jpeg');
res.write(data);
res.end();
})
}
}
然後我們在同級目錄下放一個圖片試一下:
數據持久化
數據持久化的方式有好幾種,一般都是存資料庫,少數情況下也有存文件的。存資料庫比較麻煩,還需要創建和連接資料庫,我們這裡不好demo
,我們這裡演示一個存文件的例子。一般POST
請求是用來存新數據的,我們在前面的基礎上再添加一個POST /api/users
來新增一條數據,只需要在前面的if (method === 'GET')
後面加一個POST
的判斷就行:
// ... 省略其他程式碼 ...
else if (method === 'POST') {
// 注意數據傳過來可能有多個chunk
// 我們需要拼接這些chunk
let postData = '';
req.on('data', chunk => {
postData = postData + chunk;
})
req.on('end', () => {
// 數據傳完後往db.txt插入內容
fs.appendFile(path.join(__dirname, 'db.txt'), postData, () => {
res.end(postData); // 數據寫完後將數據再次返回
});
})
}
然後我們測試一下這個API
:
再去看看文件裡面寫進去沒有:
總結
到這裡我們就完成了一個具有基本功能的web伺服器
,程式碼不複雜,但是對於幫我們理解Node web伺服器
的原理很有幫助。但是上述程式碼還有個很大的問題就是:程式碼很醜!所有程式碼都寫在一堆,而且HTTP動詞
和路由匹配全部是使用if
條件判斷,如果有幾百個API
,再配合十來個動詞,那程式碼簡直就是個災難!所以我們應該將路由處理
,HTTP動詞
,靜態文件
,數據持久化
這些功能全部抽離出來,讓整個應用變得更優雅,更好擴展。這就是Express
和Koa
這些框架存在的意義,下一篇文章我們就去Express
的源碼看看他是怎麼解決這個問題的,點個關注不迷路~
本文可運行程式碼示例已經上傳GitHub,大家可以拿下來玩玩://github.com/dennis-jiang/Front-End-Knowledges/tree/master/Examples/Node.js/HttpServer
文章的最後,感謝你花費寶貴的時間閱讀本文,如果本文給了你一點點幫助或者啟發,請不要吝嗇你的贊和GitHub小星星,你的支援是作者持續創作的動力。
作者博文GitHub項目地址: //github.com/dennis-jiang/Front-End-Knowledges
作者掘金文章匯總://juejin.im/post/5e3ffc85518825494e2772fd
我也搞了個公眾號[進擊的大前端],不打廣告,不寫水文,只發高品質原創,歡迎關注~