深入解析nodejs HTTP服務(wù)
我最近在研究nodejs的路上,正好這兩天了解了nodejs HTTP服務(wù),那么今天也算個(gè)學(xué)習(xí)筆記吧!
nodejs最重要的方面之一是具有非常迅速的實(shí)現(xiàn)HTTP和HTTPS服務(wù)器和服務(wù)的能力。http服務(wù)是相當(dāng)?shù)蛯哟蔚?,你可能要用到不同的模塊,如express來實(shí)現(xiàn)完整的Web服務(wù)器,http模塊不提供處理路由、cookie、緩存等的調(diào)用。我們主要用http模塊的地方是實(shí)現(xiàn)供應(yīng)用程序使用的后端Web服務(wù)。
1.處理URL
統(tǒng)一資源定位符(URL)為把一個(gè)請(qǐng)求發(fā)到正確的服務(wù)器的特定端口上,并訪問合適的數(shù)據(jù)提供了所有需要的信息。一個(gè)URL可以被分解成幾個(gè)不同的組成部分,每個(gè)部分都為Web服務(wù)器如何路由和處理來自客戶端的HTTP請(qǐng)求提供一塊基本的信息。node提供了url模塊,提供了把一個(gè)URL字符串轉(zhuǎn)換成一個(gè)URL對(duì)象的功能。
要從一個(gè)URL字符串創(chuàng)建URL對(duì)象,把URL字符串作為第一個(gè)參數(shù)下面的方法:
url.parse(urlStr,[parseQueryString],[slashesDenoteHost)
其中,parseQueryString參數(shù)是一個(gè)布爾值,如果為true,那么也把URL的查詢字符串部分解析為對(duì)象字面量,默認(rèn)值為false。
slashesDenoteHost參數(shù)也是一個(gè)布爾值,如果為true,那么就會(huì)把格式為//host/path的URL解析為{host:'host',pathname:'/path'},而不是{pathname:'//host/path'}
還可以用url.format(urlObj)方法將一個(gè)URL對(duì)象轉(zhuǎn)換成字符串的形式。
由url.parse()創(chuàng)建出來的URL對(duì)象的屬性:
- path 完整路徑,包括路徑和搜索
- query 要么是查詢字符串中的參數(shù)部分,要么是含有查詢字符串參數(shù)和值的解析后的對(duì)象。如果parseQueryString設(shè)置為true,那么就是解析后的對(duì)象
- search URL的查詢字符串部分,包括前導(dǎo)的問號(hào)
- pathname URL的路徑部分(包括最初的斜線,如果存在的話)
- port 主機(jī)的端口號(hào)
- hostname 主機(jī)的主機(jī)名,小寫
- auth URL的身份認(rèn)證信息
- host URL的完整主機(jī)部分,包括端口信息,小寫
- protocol 請(qǐng)求協(xié)議
- href 這是最初解析的完整的URL字符串
- hash URL的散列部分,包括井號(hào)
2.解析URL組件
url模塊有用的一種功能就是用與瀏覽器相同的方式來解析URL的組件,這可以讓你在服務(wù)器端操作URL字符串,以在URL中作出調(diào)整。
把一個(gè)URL解析到新的位置:
url.resolve(from,to)
其中,from指定了原始基礎(chǔ)URL字符串,to指定了想要URL被解析到的新位置。
舉個(gè)例子:
var url=require('url');
var originalUrl='http://user:pass@host:80/resource/path?query=string#hash';
var newResource='/another/path?querynew';
console.log(url.resolve(originalUrl,newResource));

3.處理查詢字符串和表單參數(shù)
HTTP請(qǐng)求通常在URL中包含查詢字符串(由URL對(duì)象獲得)或在正文內(nèi)包含參數(shù)數(shù)據(jù)來處理表單的提交(從客戶端請(qǐng)求的正文讀出)。查詢字符串和表單參數(shù)都只是基本的鍵值對(duì),需要使用querystring模塊的parse()方法將字符串轉(zhuǎn)換成JavaScript對(duì)象:
querystring.parse(str,[sep],[eq],[options])
str是查詢或參數(shù)字符串,sep參數(shù)允許你指定使用的分隔符,默認(rèn)的分隔符是&,eq參數(shù)允許你指定分析時(shí)使用的賦值運(yùn)算符,默認(rèn)值為=,options參數(shù)是一個(gè)具有屬性maxKeys的對(duì)象,它能夠讓你限制生成的對(duì)象可以包含的鍵的數(shù)量,默認(rèn)是1000。
var qstring=require('querystring');
var params=qstring.parse('name=Braad&color=red&color=blue');
console.log(params);

4.請(qǐng)求、響應(yīng)和服務(wù)器對(duì)象
(1) http.ClientRequest對(duì)象
當(dāng)你構(gòu)建一個(gè)HTTP客戶端時(shí),調(diào)用http.request()使得一個(gè)ClientRequest對(duì)象在內(nèi)部被創(chuàng)建,這個(gè)對(duì)象是為了當(dāng)該請(qǐng)求在服務(wù)器上進(jìn)展的時(shí)候來表示它。通常使用ClientRequest對(duì)象來啟動(dòng)、監(jiān)控和處理來自服務(wù)器的響應(yīng)。
實(shí)現(xiàn)ClientRequest對(duì)象:
http.request(options,callback)
其中,options參數(shù)是一個(gè)對(duì)象,其屬性定義了如何把客戶端的HTTP請(qǐng)求打開并發(fā)送到服務(wù)器,包括host,path,method,port等屬性。callback是一個(gè)回調(diào)函數(shù),在把請(qǐng)求發(fā)送到服務(wù)器后,處理從服務(wù)器返回的響應(yīng)時(shí)調(diào)用此回調(diào)函數(shù),此回調(diào)函數(shù)唯一的參數(shù)是一個(gè)IncomingMessage對(duì)象,該對(duì)象是來自服務(wù)器的響應(yīng)。
一個(gè)ClientRequest對(duì)象的基本實(shí)現(xiàn):
var http=require('http');
var options={
hostname:'www.myserver.com',
path:'/',
port:'8080',
method:'POST'
};
var req=http.request(options,function(response){
var str='';
response.on('data',function(chunk){
str+=chunk;
});
response.on('end',function(){
console.log(str);
});
});
req.end();
ClientRequest對(duì)象的方法:
- write(chunk,[encoding]) 把一個(gè)正文數(shù)據(jù)塊(Buffer或String對(duì)象)寫入請(qǐng)求
- end([data],[encoding]) 把可選的數(shù)據(jù)寫入請(qǐng)求正文,然后刷新Writeable流并終止該請(qǐng)求
- abort() 終止當(dāng)前的請(qǐng)求
- setTimeout(timeout,[callback]) 為請(qǐng)求設(shè)置套接字超時(shí)時(shí)間
- setNoDelay([noDelay]) 禁用在發(fā)送數(shù)據(jù)之前緩沖數(shù)據(jù)的Nagle算法,noDelay是一個(gè)布爾值,true表示立即寫,false表示緩沖寫入
- setSocketKeepAlive([enable],[initialDelay]) 啟用和禁用對(duì)客戶機(jī)請(qǐng)求的保持活動(dòng)功能,enable默認(rèn)為false,即禁用;initialDelay指定最后一個(gè)數(shù)據(jù)包和第一個(gè)保持活動(dòng)請(qǐng)求之間的延遲
(2)http.ServerResponse對(duì)象
當(dāng)HTTP服務(wù)器接收到一個(gè)Request事件時(shí),它在內(nèi)部創(chuàng)建ServerResponse對(duì)象,這個(gè)對(duì)象作為第二個(gè)參數(shù)被傳遞到request事件處理程序??梢允褂肧erverResponse對(duì)象指定并發(fā)送到客戶端的響應(yīng)。
ServerResponse對(duì)象的方法:
- writeContinue() 發(fā)送一個(gè)HTTP/1.1 100 Continue消息給客戶端,請(qǐng)求被發(fā)送的正文內(nèi)容
- writeHead(statusCode,[reasonPhrase],[headers]) 把一個(gè)響應(yīng)標(biāo)頭寫入請(qǐng)求,例如response.writeHead(200,'successs',{'Content-Length':body.length,'Content-Type':'text/plain'});
- setTimeout(timeout,[callback]) 設(shè)置客戶端連接的套接字超時(shí)時(shí)間
- setHeader(name,value) 設(shè)置一個(gè)特定的標(biāo)頭值
- getHeader(name) 獲取已在響應(yīng)中設(shè)置的一個(gè)HTTP標(biāo)頭
- removeHeader(name) 移除已在響應(yīng)中設(shè)置的一個(gè)HTTP標(biāo)頭
- write(chunk,[encoding]) 寫入chunk,Buffer或String對(duì)象到響應(yīng)Writeable流,僅把數(shù)據(jù)寫入響應(yīng)的正文部分
- addTrailers(headers) 將HTTP尾隨標(biāo)頭寫入響應(yīng)的結(jié)束處
- end([data],[encoding]) 把可選的數(shù)據(jù)輸出寫入響應(yīng)的正文,然后刷新Writeable流并響應(yīng)該請(qǐng)求
(3)http.IncomingMessage對(duì)象
HTTP服務(wù)器和客戶端都創(chuàng)建IncomingMessage對(duì)象,該對(duì)象實(shí)現(xiàn)了Readable流,讓你能夠把客戶端請(qǐng)求或服務(wù)器響應(yīng)作為流院讀入,這意味著它們的readable和data事件可以被監(jiān)聽并用來從流中讀出數(shù)據(jù)。
IncomingMessage對(duì)象中可用的事件、屬性和方法
- close 當(dāng)?shù)讓犹捉幼直魂P(guān)閉時(shí)發(fā)出
- httpVersion 指定用于構(gòu)建客戶端請(qǐng)求/響應(yīng)的HTTP版本
- headers 包含了隨請(qǐng)求/響應(yīng)發(fā)送的標(biāo)頭的一個(gè)對(duì)象
- method 指定用于請(qǐng)求/響應(yīng)的方法
- url 發(fā)送到服務(wù)器的URL字符串
- statusCode 指定來自服務(wù)器的3位數(shù)狀態(tài)碼
- socket 一個(gè)指向net.Socket對(duì)象的句柄,用來與客戶端/服務(wù)器的通信
- setTimeout(timeout,[callback]) 設(shè)置連接的套接字超時(shí)時(shí)間
(4)HTTP Server對(duì)象
HTTP Server對(duì)象提供了實(shí)現(xiàn)HTTP服務(wù)器的基本框架,它提供了一個(gè)監(jiān)聽端口的底層套接字和接收請(qǐng)求,然后發(fā)送響應(yīng)給客戶端連接的處理程序,當(dāng)服務(wù)器正在監(jiān)聽時(shí),node應(yīng)用程序并沒有結(jié)束。
要啟動(dòng)HTTP服務(wù)器,首先需要createServer()方法創(chuàng)建一個(gè)Server對(duì)象:
http.createServer([requestListener])
此方法返回Server對(duì)象,requestListener參數(shù)是在請(qǐng)求事件被觸發(fā)時(shí)執(zhí)行的回調(diào)函數(shù),此回調(diào)函數(shù)接收兩個(gè)參數(shù),一個(gè)是代表客戶端請(qǐng)求的IncomingMessage對(duì)象,第二個(gè)參數(shù)是用來指定和發(fā)送響應(yīng)的ServerResponse對(duì)象。
創(chuàng)建了server對(duì)象,就可以在server對(duì)象上的listen()方法開始監(jiān)聽它:
listen(port,[hostname],[backlog],[callback])
一個(gè)基本的例子:
var http=require('http');
http.createServer(function(req,res){
//handle resquest and response
}).listen(8080);
實(shí)現(xiàn)一個(gè)基本的靜態(tài)文件服務(wù)
//實(shí)現(xiàn)一個(gè)靜態(tài)文件服務(wù)器
var fs=require('fs');
var http=require('http');
var url=require('url');
//創(chuàng)建一個(gè)服務(wù)器
http.createServer(function(req,res){
if(req.url!="/favicon.ico"){
var urlObj=url.parse(req.url,true,false);
console.log(urlObj.pathname);
fs.readFile('.'+urlObj.pathname+'.html',function(err,data){
if(err){
res.writeHead(404);
res.end(JSON.stringify(err));
return;
}
console.log(data.toString());
//將文件的內(nèi)容寫入res響應(yīng)對(duì)象
res.end(data);
});
}
}).listen(8080);
//實(shí)現(xiàn)一個(gè)HTTP客戶端,向服務(wù)器發(fā)送一個(gè)GET請(qǐng)求來檢索文件內(nèi)容
var options={
hostname:'127.0.0.1',
port:'8080',
path:'/aaa'
};
//on('data')讀取來自服務(wù)器的響應(yīng)中的內(nèi)容,on('end')把文件內(nèi)容記錄到到一個(gè)文件
function handleResponse(response){
var serverData='';
response.on('data',function(chunk){
serverData+=chunk;
});
response.on('end',function(){
console.log(serverData);
});
}
http.request(options,function(response){
handleResponse(response);
}).end();

實(shí)現(xiàn)動(dòng)態(tài)的GET服務(wù)器
var http=require('http');
var messages=[
'message1',
'message2',
'message3'
];
http.createServer(function(req,res){
res.setHeader('Content-Type','text/html');
res.writeHead(200);
res.write('<html><head><title>HTTP Server</title></head>');
res.write('<body>');
for(var idx in messages){
res.write('\n<h1>'+messages[idx]+'</h1>');
}
res.end('\n</body></html>');
}).listen(8080);
var options={
hostname:'localhost',
port:'8080'
};
function handleResponse(response){
var serverData='';
response.on('data',function(chunk){
serverData+=chunk;
});
response.on('end',function(){
console.log('response status: ',response.statusCode);
console.log('response headers: ',response.headers);
console.log(serverData);
});
}
http.request(options,function(response){
handleResponse(response);
}).end();

以上就是本文的全部?jī)?nèi)容,希望對(duì)大家的學(xué)習(xí)有所幫助,也希望大家多多支持腳本之家。
- NodeJS學(xué)習(xí)筆記之Http模塊
- nodejs使用http模塊發(fā)送get與post請(qǐng)求的方法示例
- 深入淺析Nodejs的Http模塊
- nodejs利用http模塊實(shí)現(xiàn)銀行卡所屬銀行查詢和騷擾電話驗(yàn)證示例
- nodejs實(shí)現(xiàn)HTTPS發(fā)起POST請(qǐng)求
- 詳解nodejs通過代理(proxy)發(fā)送http請(qǐng)求(request)
- 使用nodejs、Python寫的一個(gè)簡(jiǎn)易HTTP靜態(tài)文件服務(wù)器
- NodeJS搭建HTTP服務(wù)器的實(shí)現(xiàn)步驟
- 搭建簡(jiǎn)單的nodejs http服務(wù)器詳解
- NodeJS創(chuàng)建最簡(jiǎn)單的HTTP服務(wù)器
- nodejs 使用http進(jìn)行post或get請(qǐng)求的實(shí)例(攜帶cookie)
- NodeJS http模塊用法示例【創(chuàng)建web服務(wù)器/客戶端】
相關(guān)文章
NodeJS父進(jìn)程與子進(jìn)程資源共享原理與實(shí)現(xiàn)方法
這篇文章主要介紹了NodeJS父進(jìn)程與子進(jìn)程資源共享原理與實(shí)現(xiàn)方法,結(jié)合實(shí)例形式分析了nodejs基于cluster模塊實(shí)現(xiàn)父進(jìn)程與子進(jìn)程資源共享的相關(guān)操作技巧,需要的朋友可以參考下2018-03-03
手寫Node靜態(tài)資源服務(wù)器的實(shí)現(xiàn)方法
這篇文章主要介紹了手寫Node靜態(tài)資源服務(wù)器的實(shí)現(xiàn)方法,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2018-03-03
在Node.js中使用Swagger自動(dòng)生成API接口文檔
這篇文章主要給大家介紹了如何在Node.js項(xiàng)目中使用 Swagger 來自動(dòng)生成 API接口文檔,使用生成方式有很多種,本文基于swagger-jsdoc+swagger-ui-express快速實(shí)現(xiàn),文中通過代碼示例介紹的非常詳細(xì),需要的朋友可以參考下2024-01-01
深入了解 Node的多進(jìn)程服務(wù)實(shí)現(xiàn)
本文主要介紹了Node的多進(jìn)程服務(wù)實(shí)現(xiàn),文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2022-06-06

