国产探花免费观看_亚洲丰满少妇自慰呻吟_97日韩有码在线_资源在线日韩欧美_一区二区精品毛片,辰东完美世界有声小说,欢乐颂第一季,yy玄幻小说排行榜完本

首頁 > 開發(fā) > JS > 正文

NodeJS搭建HTTP服務器的實現(xiàn)步驟

2024-05-06 16:46:23
字體:
供稿:網(wǎng)友

前言

在 NodeJS 中用來創(chuàng)建服務的模塊是 http 核心模塊,本篇就來介紹關(guān)于使用 http 模塊搭建 HTTP 服務器和客戶端的方法,以及模塊的基本 API。

HTTP 服務器

1、創(chuàng)建 HTTP 服務器

在 NodeJS 中,創(chuàng)建 HTTP 服務器可以與 net 模塊創(chuàng)建 TCP 服務器對比,創(chuàng)建服務器有也兩種方式。

方式 1:

const http = require("http");const server = http.createServer(function(req, res) {  // ......});server.listen(3000);

方式 2:

const http = require("http");const server = http.createServer();server.on("request", function(req, res) {  // ......});server.listen(3000);

在 createServer 的回調(diào)和 request 事件的回調(diào)函數(shù)中有兩個參數(shù),req(請求)、res(響應),基于 socket,這兩個對象都是 Duplex 類型的可讀可寫流。

http 模塊是基于 net 模塊實現(xiàn)的,所以 net 模塊原有的事件在 http 中依然存在。

const http = require("http");const server = http.createServer();// net 模塊事件server.on("connection", function(socket) {  console.log("連接成功");});server.listen(3000);

2、獲取請求信息

在請求對象 req 中存在請求的方法、請求的 url(包含參數(shù),即查詢字符串)、當前的 HTTP 協(xié)議版本和請求頭等信息。

const http = require("http");const server = http.createServer();server.on("request", function(req, res) {  console.log(req.method); // 獲取請求方法  console.log(req.url); // 獲取請求路徑(包含查詢字符串)  console.log(req.httpVersion); // 獲取 HTTP 協(xié)議版本  console.log(req.headers); // 獲取請求頭(對象)  // 獲取請求體的內(nèi)容  let arr = [];  req.on("data", function(data) {    arr.push(data);  });  req.on("end", function() {    console.log(Buffer.concat(arr).toString());  });});server.listen(3000, function() {  console.log("server start 3000");});

通過 req 對應的屬性可以拿到請求行和請求首部的信息,請求體內(nèi)的內(nèi)容通過流操作來獲取,其中 url 中存在多個有用的參數(shù),我們自己處理會很麻煩,可以通過 NodeJS 的核心模塊 url 進行解析。

const url = require("url");let str = "http://user:pass@www.pandashen.com:8080/src/index.html?a=1&b=2#hash";// parse 方法幫助我們解析 url 路徑let obj = url.parse(str, true);console.log(obj);// {//   protocol: 'http:',//   slashes: true,//   auth: 'user:pas',//   host: 'www.pandashen.com:8080',//   port: '8080',//   hostname: 'www.pandashen.com',//   hash: '#hash',//   search: '?a=1&b=2',//   query: '{ a: '1', b: '2' }',//   pathname: '/src/index.html'//   path: '/src/index.html?a=1&b=2',//   href: 'http://user:pass@www.pandashen.com:8080/src/index.html?a=1&b=2#hash' }

在被解析路徑返回的對象中有幾個屬性被經(jīng)常使用:

  • host:主機(域名 + 端口號);
  • hostname:主機名;
  • query:請求參數(shù)(查詢字符串或參數(shù)對象);
  • pathname:資源路徑(根據(jù)不同的路徑返回不同的資源)。

我們使用 url 的 parse 方法來幫我們解析請求路徑,在真實的服務器中傳入的第一個參數(shù)為 req.url,第二個參數(shù)不傳時,query 會被解析成 a=1&b=2 的形式,第二個參數(shù)傳入 true,query 屬性的查詢字符串會被解析成對象的形式。

url 模塊中,將查詢字符串 a=1&b=2 轉(zhuǎn)換為對象 { a: '1', b: '2' } 的實現(xiàn)方式其實是使用正則替換實現(xiàn)的。

模擬查詢字符串轉(zhuǎn)換對象的核心邏輯:

let str = "a=1&b=2&c=3";let obj = {};str.replace(/([^=&]+)=([^=&]+)/g, function() {  obj[arguments[1]] = arguments[2];});console.log(obj); // { a: '1', b: '2', c: '3' }

在上面代碼的 replace 方法的回調(diào)函數(shù)中參數(shù)集合的第一項為匹配到的字符串,第二項為第一個分組的值,第三項為第二個分組的值,依次類推,倒數(shù)第二項為分組匹配的索引,最后一項為原字符串。

3、設(shè)置響應信息

我們可以通過 req 來獲取請求信息,自然也可以通過 res 來設(shè)置響應信息返回給客戶端。

const http = require("http");const server = http.createServer();server.on("request", function(req, res) {  // 設(shè)置響應頭(過去的用法),不能多次調(diào)用,見到要認識  res.writeHead(200, { "Content-Type": "text", a: "hello world" });  // 設(shè)置響應頭(現(xiàn)在的用法,常用),可以多次調(diào)用,每次設(shè)置一個響應頭  res.setHeader("Content-Type", "text");  // 設(shè)置狀態(tài)碼,不設(shè)置默認為 200  res.statusCode = 200;  // 不發(fā)送 Date(日期)響應頭  res.sendDate = false;  // 返回內(nèi)容  res.write("hello world"); // 不會關(guān)閉連接  res.end("hello world"); // 將內(nèi)容返回后關(guān)閉連接});server.listen(3000, function() {  console.log("server start 3000");});

返回給客戶端的信息主要分為兩部分,分別為響應頭和返回給瀏覽器的內(nèi)容,在不設(shè)置響應頭的情況下,默認會設(shè)置響應頭 Content-Length 和 Date ,代表當前返回給客戶端的內(nèi)容長度和日期。

返回給瀏覽器的內(nèi)容可以通過 res 的 write 方法和 end 方法進行發(fā)送,write 方法不會斷開連接(通常在響應后需要斷開與客戶端的連接),end 方法會斷開連接,在 end 方法存在參數(shù)時,會在內(nèi)部調(diào)用 write 將參數(shù)內(nèi)容返回給客戶端,并斷開連接。

HTTP 客戶端

在 net 模塊中可以通過 net.createConnection 來創(chuàng)建客戶端,并發(fā)送請求到服務端,在 http 模塊同樣可以創(chuàng)建客戶端,并向 http 服務器發(fā)送請求。

// 客戶端:client.jsconst http = require("http");// 發(fā)送請求的配置let config = {  host: "localhost",  port: 3000,  method: "get",  headers: {    a: 1  }};// 創(chuàng)建客戶端let client = http.request(config, function(res) {  // 接收服務端返回的數(shù)據(jù)  let arr = [];  res.on("data", function(data) {    arr.push(data);  });  res.on("end", function() {    console.log(Buffer.concat(arr).toString());  });});// 發(fā)送請求client.end();

在 http 模塊中通過 request 方法創(chuàng)建客戶端,該方法第一個參數(shù)為發(fā)送請求的配置,包含請求地址、端口號、請求方法以及請求頭等,第二個參數(shù)為回調(diào)函數(shù),在請求被響應后執(zhí)行,回調(diào)函數(shù)的參數(shù)為服務器的響應對象 res,創(chuàng)建的客戶端通過 end 方法將請求發(fā)出與服務端進行通信。

使用 NodeJS 實現(xiàn)的 “爬蟲” 其實就可以通過 http 模塊創(chuàng)建的客戶端來實現(xiàn),客戶端幫我們向我們要抓取數(shù)據(jù)的地址發(fā)送請求,并拿到響應的數(shù)據(jù)進行解析。

同時使用 HTTP 客戶端和服務器

我們使用自己創(chuàng)建的客戶端訪問自己的服務端,并體會請求響應的過程,就是用上面 client.js 作為客戶端,啟動 server.js 后再啟動 client.js 查看效果。

// 服務器:server.jsconst http = require("http");http.createServer(function(req, res) {  console.log("The request came");  // 獲取客戶端請求信息  console.log(req.method);  console.log(req.headers);  // 返回數(shù)據(jù)  res.write("hello world");}).listen(3000, function() {  console.log("server start 3000");});

簡易爬蟲

我們結(jié)合 http 模塊創(chuàng)建的服務端和客戶端實現(xiàn)一個簡易版的 “爬蟲” 去抓取百度新聞頁所有 li 標簽內(nèi)的文章標題。

// 簡易爬蟲:crawl.jsconst http = require("http");// 創(chuàng)建服務器const server = http.createServer();// 監(jiān)聽請求server.on("request", function(req, res) {  let client = http.request(    {      host: "news.baidu.com",      method: "get",      port: 80    },    function(r) {      // 接收百度新聞返回的數(shù)據(jù)      let arr = [];      r.on("data", function(data) {        arr.push(data);      });      r.on("end", function() {        // 處理數(shù)據(jù)        let result = Buffer.concat(arr).toString();        let matches = result.match(/<li class="bold-item">([/s/S*?])<//li>/gm);        // 設(shè)置返回給瀏覽器的文檔類型和編碼格式        res.setHeader("Content-Type", "text/html;charset=utf8");        // 響應瀏覽器        res.end(matches.join(""));      });    }  );  client.end();});server.listen(3000);

上面的正則匹配中 ([/s/S*?]) 代表匹配 <li class="bold-item"> 到 <//li> 之間所有內(nèi)容(多個字符、非貪婪模式),gm 代表全局并多行匹配。

上面爬取百度新聞數(shù)據(jù)的過程中,我們自己的 Node 服務器扮演了一個 “中間層” 的角色,我們通過瀏覽器訪問自己的服務器 localhost:3000 觸發(fā) request 事件,執(zhí)行了回調(diào),在回調(diào)中創(chuàng)建客戶端向 news.baidu.com 發(fā)送了請求,并在客戶端的回調(diào)中處理了響應(百度新聞頁返回的數(shù)據(jù)),將處理后的內(nèi)容通過我們自己 Node 服務器的 res 對象返回給了瀏覽器。

總結(jié)

相信在讀過本篇文章之后對搭建一個 Node 服務應該已經(jīng)有了思路,為未來通過 Node 服務實現(xiàn)復雜的業(yè)務場景及數(shù)據(jù)的處理打下了一個基礎(chǔ),希望初學 Node 的小伙伴在看了這篇文章后能有所收獲。

以上就是本文的全部內(nèi)容,希望對大家的學習有所幫助,也希望大家多多支持VeVb武林網(wǎng)。


注:相關(guān)教程知識閱讀請移步到JavaScript/Ajax教程頻道。
發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
主站蜘蛛池模板: 湟中县| 深水埗区| 阜南县| 丰台区| 招远市| 光泽县| 革吉县| 大连市| 上犹县| 河曲县| 重庆市| 伊通| 东乡县| 泸溪县| 松阳县| 团风县| 当雄县| 姜堰市| 商丘市| 阿图什市| 潜江市| 阳新县| 永福县| 古田县| 界首市| 临湘市| 汉中市| 马边| 阳曲县| 绥棱县| 婺源县| 吉水县| 新巴尔虎右旗| 焦作市| 中阳县| 怀宁县| 白朗县| 恭城| 辉县市| 古蔺县| 平塘县|