這篇文章主要介紹了Nodejs中Express 常用中間件body-parser 實作解析,具有一定的參考價值,有興趣的小夥伴可以參考一下
##寫在前面
body-parser是非常常用的一個
express中間件,作用是對
post請求的請求體進行解析。使用非常簡單,以下兩行程式碼已經涵蓋了大部分的使用場景。
app.use(bodyParser.json()); app.use(bodyParser.urlencoded({ extended: false }));本文從簡單的例子出發,探究
body-parser的內部實作。至於
body-parser如何使用,有興趣的同學可以參考官方文件。
入門基礎
在正式講解前,我們先來看一個POST請求的報文,如下所示。POST /test HTTP/1.1 Host: 127.0.0.1:3000 Content-Type: text/plain; charset=utf8 Content-Encoding: gzip chyingp其中需要我們注意的有
Content-Type、
Content-Encoding以及封包主體:
字串chyingp。
body-parser主要做了什麼
#body-parser實作的重點如下:
一、處理不同類型請求體
為了方便讀者測試,以下範例都包含服務端、客戶端程式碼,完整程式碼可在筆者github上找到。解析text/plain
客戶端請求的程式碼如下,採用預設編碼,不對請求體進行壓縮。請求體類型為text/plain。
var http = require('http'); var options = { hostname: '127.0.0.1', port: '3000', path: '/test', method: 'POST', headers: { 'Content-Type': 'text/plain', 'Content-Encoding': 'identity' } }; var client = http.request(options, (res) => { res.pipe(process.stdout); }); client.end('chyingp');服務端程式碼如下。
text/plain類型處理比較簡單,就是buffer的拼接。
var http = require('http'); var parsePostBody = function (req, done) { var arr = []; var chunks; req.on('data', buff => { arr.push(buff); }); req.on('end', () => { chunks = Buffer.concat(arr); done(chunks); }); }; var server = http.createServer(function (req, res) { parsePostBody(req, (chunks) => { var body = chunks.toString(); res.end(`Your nick is ${body}`) }); }); server.listen(3000);
解析application/json
客戶程式碼如下,把Content-Type換成
application/json。
var http = require('http'); var querystring = require('querystring'); var options = { hostname: '127.0.0.1', port: '3000', path: '/test', method: 'POST', headers: { 'Content-Type': 'application/json', 'Content-Encoding': 'identity' } }; var jsonBody = { nick: 'chyingp' }; var client = http.request(options, (res) => { res.pipe(process.stdout); }); client.end( JSON.stringify(jsonBody) );服務端程式碼如下,相較於
text/plain,只是多了個
JSON.parse()的過程。
var http = require('http'); var parsePostBody = function (req, done) { var length = req.headers['content-length'] - 0; var arr = []; var chunks; req.on('data', buff => { arr.push(buff); }); req.on('end', () => { chunks = Buffer.concat(arr); done(chunks); }); }; var server = http.createServer(function (req, res) { parsePostBody(req, (chunks) => { var json = JSON.parse( chunks.toString() ); // 关键代码 res.end(`Your nick is ${json.nick}`) }); }); server.listen(3000);
解析application/x-www-form-urlencoded
客戶端程式碼如下,這裡透過querystring對請求體進行格式化,得到類似
nick=chyingp的字串。
var http = require('http'); var querystring = require('querystring'); var options = { hostname: '127.0.0.1', port: '3000', path: '/test', method: 'POST', headers: { 'Content-Type': 'form/x-www-form-urlencoded', 'Content-Encoding': 'identity' } }; var postBody = { nick: 'chyingp' }; var client = http.request(options, (res) => { res.pipe(process.stdout); }); client.end( querystring.stringify(postBody) );服務端程式碼如下,同樣跟
text/plain的解析差不多,就多了個
querystring.parse()的呼叫。
var http = require('http'); var querystring = require('querystring'); var parsePostBody = function (req, done) { var length = req.headers['content-length'] - 0; var arr = []; var chunks; req.on('data', buff => { arr.push(buff); }); req.on('end', () => { chunks = Buffer.concat(arr); done(chunks); }); }; var server = http.createServer(function (req, res) { parsePostBody(req, (chunks) => { var body = querystring.parse( chunks.toString() ); // 关键代码 res.end(`Your nick is ${body.nick}`) }); }); server.listen(3000);
二、處理不同編碼
很多時候,來自客戶端的請求,採用的不一定是預設的utf8編碼,這個時候,就需要對請求體進行解碼處理。
var http = require('http'); var iconv = require('iconv-lite'); var encoding = 'gbk'; // 请求编码 var options = { hostname: '127.0.0.1', port: '3000', path: '/test', method: 'POST', headers: { 'Content-Type': 'text/plain; charset=' + encoding, 'Content-Encoding': 'identity', } }; // 备注:nodejs本身不支持gbk编码,所以请求发送前,需要先进行编码 var buff = iconv.encode('程序猿小卡', encoding); var client = http.request(options, (res) => { res.pipe(process.stdout); }); client.end(buff, encoding);服務端程式碼如下,這裡多了兩個步驟:編碼判斷、解碼操作。首先透過
Content-Type取得編碼類型
gbk,然後透過
iconv-lite進行反向解碼操作。
var http = require('http'); var contentType = require('content-type'); var iconv = require('iconv-lite'); var parsePostBody = function (req, done) { var obj = contentType.parse(req.headers['content-type']); var charset = obj.parameters.charset; // 编码判断:这里获取到的值是 'gbk' var arr = []; var chunks; req.on('data', buff => { arr.push(buff); }); req.on('end', () => { chunks = Buffer.concat(arr); var body = iconv.decode(chunks, charset); // 解码操作 done(body); }); }; var server = http.createServer(function (req, res) { parsePostBody(req, (body) => { res.end(`Your nick is ${body}`) }); }); server.listen(3000);
三、處理不同壓縮型別
這裡舉個gzip壓縮的例子。客戶端程式碼如下,重點如下:
var http = require('http'); var zlib = require('zlib'); var options = { hostname: '127.0.0.1', port: '3000', path: '/test', method: 'POST', headers: { 'Content-Type': 'text/plain', 'Content-Encoding': 'gzip' } }; var client = http.request(options, (res) => { res.pipe(process.stdout); }); // 注意:将 Content-Encoding 设置为 gzip 的同时,发送给服务端的数据也应该先进行gzip var buff = zlib.gzipSync('chyingp'); client.end(buff);服務端程式碼如下,這裡透過
zlib模組,對請求體進行了解壓縮操作(guzip)。
var http = require('http'); var zlib = require('zlib'); var parsePostBody = function (req, done) { var length = req.headers['content-length'] - 0; var contentEncoding = req.headers['content-encoding']; var stream = req; // 关键代码如下 if(contentEncoding === 'gzip') { stream = zlib.createGunzip(); req.pipe(stream); } var arr = []; var chunks; stream.on('data', buff => { arr.push(buff); }); stream.on('end', () => { chunks = Buffer.concat(arr); done(chunks); }); stream.on('error', error => console.error(error.message)); }; var server = http.createServer(function (req, res) { parsePostBody(req, (chunks) => { var body = chunks.toString(); res.end(`Your nick is ${body}`) }); }); server.listen(3000);
寫在後面
body-parser的核心實作並不複雜,翻看原始碼後你會發現,更多的程式碼是在處理異常跟邊界。
Content-Type是
multipart/form-data,這個的處理相對複雜些,
body-parser不打算對其進行支援。篇幅有限,後續章節再繼續展開。
#5. JavaScript表單驗證實作程式碼_javascript技巧
以上是Nodejs中常用中間件body-parser的實例詳解的詳細內容。更多資訊請關注PHP中文網其他相關文章!