如何在nodejs中體驗(yàn)http/2詳解
前言
2015年,HTTP/2 發(fā)布,直到2021年公司的項(xiàng)目才開始在實(shí)踐中應(yīng)用;自己對(duì)http2諸多特點(diǎn)的理解只存在于字面上,于是嘗試在nodejs中實(shí)踐一下,加深自己的理解。
多路復(fù)用
同域名下所有通信都在單個(gè)連接上完成,消除了因多個(gè) TCP 連接而帶來的延時(shí)和內(nèi)存消耗,這在大量請(qǐng)求同時(shí)發(fā)出的情況下能夠減少加載時(shí)間。
使用如下代碼查看http2環(huán)境下,資源下載的情況(瀏覽器開啟限流和disable cache):
const http2 = require('http2'); const fs = require('fs'); const { HTTP2_HEADER_PATH } = http2.constants; const server = http2.createSecureServer({ key: fs.readFileSync('localhost-privkey.pem'), cert: fs.readFileSync('localhost-cert.pem') }); server.on('error', (err) => console.error(err)); server.on('stream', (stream, headers) => { // stream is a Duplex const path = headers[':path']; if(path === '/img.png' || path === '/favicon.ico'){ const fd = fs.openSync('img.png', 'r'); const stat = fs.fstatSync(fd); const headers = { 'content-length': stat.size, 'last-modified': stat.mtime.toUTCString(), 'content-type': 'image/png' }; stream.respondWithFD(fd, headers); } else if(path === '/') { stream.respond({ 'content-type': 'text/html; charset=utf-8', ':status': 200 }); stream.end(` <h1>Hello World</h1> <script> for(var i=0;i<50;i++){ fetch('/img.png') } </script> `); } }); server.listen(8443);
可以看到當(dāng)資源開始同時(shí)請(qǐng)求,所有的請(qǐng)求形成一個(gè)隊(duì)列,請(qǐng)求之間開始時(shí)間相差大概1ms, 因?yàn)橄螺d的是同一個(gè)圖片,50張圖片同時(shí)下載,最后幾乎在同時(shí)完成下載。
下面是http1.1的例子,通過對(duì)比發(fā)現(xiàn)瀏覽器按照自己的最大并發(fā)量同時(shí)發(fā)出請(qǐng)求,只有當(dāng)請(qǐng)求返回后才發(fā)出新的請(qǐng)求(瀏覽器開啟限流和disable cache):
const http = require('http'); const fs = require('fs'); const server = http.createServer(function(req,res){ const path = req.url; if(path === '/img.png' || path === '/favicon.ico'){ res.writeHead(200,{'Content-type':'image/png'}) var stream = fs.createReadStream('img.png') stream.pipe(res) } else { res.writeHead(200, { 'Content-Type': 'text/html; charset=utf-8' }); res.end(` <h1>Hello World</h1> <script> for(var i=0;i<50;i++){ fetch('/img.png') } </script> `); } }); server.listen(8444);
服務(wù)端推送
按照如下代碼測(cè)試
const http2 = require('http2'); const fs = require('fs'); const { HTTP2_HEADER_PATH } = http2.constants; const server = http2.createSecureServer({ key: fs.readFileSync('localhost-privkey.pem'), cert: fs.readFileSync('localhost-cert.pem') }); server.on('error', (err) => console.error(err)); server.on('stream', (stream, headers) => { const path = headers[':path']; if(path === '/') { stream.respond({ 'content-type': 'text/html; charset=utf-8', ':status': 200 }); stream.pushStream({ [HTTP2_HEADER_PATH]: '/style.css' }, (err, pushStream, headers) => { if (err) throw err; const fd = fs.openSync('style.css', 'r'); const stat = fs.fstatSync(fd); const header = { 'content-length': stat.size, 'last-modified': stat.mtime.toUTCString(), 'content-type': 'text/css' }; pushStream.respondWithFD(fd, header) }); stream.end(` <h1>Hello World</h1> <script> setTimeout(()=>{ fetch('/style.css') },2000) </script> `); } else if(path === '/style.css'){ const fd = fs.openSync('style.css', 'r'); const stat = fs.fstatSync(fd); const headers = { 'content-length': stat.size, 'last-modified': stat.mtime.toUTCString(), 'content-type': 'text/css' }; stream.respondWithFD(fd, headers); } }); server.listen(8442);
資源加載情況如下,style.css的Initiator是Push,大小是66 B, 同時(shí)首頁加載的大小是207 B,
注釋掉stream.pushStream
部分后,不使用推送,資源加載如下,style.css大小是89B, 同時(shí)首頁加載的大小是182B,
綜上所看,服務(wù)端推送可以提前加載資源,優(yōu)化非首頁加載有益。
令人高興的是,因?yàn)槭褂寐实?,chrome在105版本后不再支持http2的服務(wù)端推送,導(dǎo)致這個(gè)特點(diǎn)在前端開發(fā)中可以忽略了。并且如果要測(cè)試改特點(diǎn)需要使用低版本的chrome,比如本例子使用的是chrome 96 mac版本。
本文所用代碼:https://github.com/blank-x/pg/tree/master/http2,nodejs版本是v16.19.0.
總結(jié)
到此這篇關(guān)于如何在nodejs中體驗(yàn)http/2的文章就介紹到這了,更多相關(guān)nodejs中體驗(yàn)http/2內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
用C/C++來實(shí)現(xiàn) Node.js 的模塊(一)
這篇文章的主要內(nèi)容其實(shí)簡(jiǎn)而言之就是——用C/C++來實(shí)現(xiàn) Node.js 的模塊,非常的不錯(cuò),有需要的朋友可以參考下2014-09-09修改Nodejs內(nèi)置的npm默認(rèn)配置路徑方法
今天小編就為大家分享一篇修改Nodejs內(nèi)置的npm默認(rèn)配置路徑方法,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-05-05Node.js讀寫文件之批量替換圖片的實(shí)現(xiàn)方法
下面小編就為大家?guī)硪黄狽ode.js讀寫文件之批量替換圖片的實(shí)現(xiàn)方法。小編覺得挺不錯(cuò)的,現(xiàn)在就分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2016-09-09node.js中使用node-schedule實(shí)現(xiàn)定時(shí)任務(wù)實(shí)例
這篇文章主要介紹了node.js中使用node-schedule實(shí)現(xiàn)定時(shí)任務(wù)實(shí)例,包括安裝方法和4種使用例子,需要的朋友可以參考下2014-06-06NodeJs內(nèi)存占用過高的排查實(shí)戰(zhàn)記錄
這篇文章主要給大家介紹了關(guān)于NodeJs內(nèi)存占用過高的排查實(shí)戰(zhàn)記錄,文中通過示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2021-05-05NodeJS與HTML5相結(jié)合實(shí)現(xiàn)拖拽多個(gè)文件上傳到服務(wù)器的實(shí)現(xiàn)方法
這篇文章主要介紹了NodeJS與HTML5相結(jié)合實(shí)現(xiàn)拖拽多個(gè)文件上傳到服務(wù)器的實(shí)現(xiàn)方法的相關(guān)資料,非常不錯(cuò),具有參考借鑒價(jià)值,需要的朋友可以參考下2016-07-07nodejs實(shí)現(xiàn)超簡(jiǎn)單生成二維碼的方法
這篇文章主要介紹了nodejs實(shí)現(xiàn)超簡(jiǎn)單生成二維碼的方法,結(jié)合實(shí)例形式分析了nodejs基于qr-image插件生成二維碼的相關(guān)操作技巧,需要的朋友可以參考下2018-03-03NodeJS 創(chuàng)建目錄和文件的方法實(shí)例分析
這篇文章主要介紹了NodeJS 創(chuàng)建目錄和文件的方法,涉及node.js中fs模塊mkdir、writeFile及目錄判斷existsSync等方法的功能與相關(guān)使用技巧,需要的朋友可以參考下2023-04-04