node網(wǎng)頁分段渲染詳解
頁面渲染,通常來說分為前端渲染以及后端渲染。前端渲染指的是服務(wù)端返回html框架以及模版,前端通過ajax異步請(qǐng)求拉取數(shù)據(jù)渲染模版,并動(dòng)態(tài)修改dom,形成最終頁面。服務(wù)端渲染則是服務(wù)端通過在后端拉取數(shù)據(jù)以及后端模版渲完整頁面,并返回到客戶端。2種方法各有好處,后端渲染帶來的則是首屏?xí)r間的提高,減少請(qǐng)求次數(shù),利于SEO等好處。但是傳統(tǒng)后端直出渲染需要等到整個(gè)網(wǎng)頁渲染完成,才能返回到客戶端。假如某個(gè)區(qū)塊拉取數(shù)據(jù)比較慢,影響了渲染的速度,那對(duì)于用戶來說,等待的時(shí)候也會(huì)跟著變長(zhǎng)對(duì)于后端渲染能否跟前端ajax渲染一樣,分塊分區(qū)域傳統(tǒng)的服務(wù)端直出渲染,下面將提供一種解決方案-網(wǎng)頁分段渲染。
首先我們先看下傳統(tǒng)的渲染方式:
const http = require("http");
const fs = require("fs");
var tpl1 = '<!DOCTYPE html><html><head><title>測(cè)試render</title></head><body>helloword<p>$data1</p>';
var tpl2 = '<p>$data2</p></body></html>';
var html = '';
var server = http.createServer((req, res)=>{
if(req.url!=="/favicon.ico"){
res.writeHead(200, {
'Content-Type' : 'text/html'
});
getDataOne((data1) => {
getDataTwo((data2) => {
res.end(tpl1.replace(/\$data1/g, data1) + tpl2.replace(/\$data2/g, data2));
})
});
}
}).listen(3000, '127.0.0.1');
function getDataOne(fn){
setTimeout(() => {
fn('11111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111');
}, 5000);
}
function getDataTwo(fn){
setTimeout(() => {
fn('22222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222');
}, 5000);
}
上面我們提供了一個(gè)簡(jiǎn)單的例子,通過訪問http://127.0.0.1:3000 返回一個(gè)頁面。其中渲染頁面時(shí),有2個(gè)耗時(shí)5秒的操作,可以假設(shè)為IO或者數(shù)據(jù)拉取。這個(gè)時(shí)候我們觀察返回頁面的時(shí)間是10秒,也就是說用戶看到頁面需要10秒鐘。

下面我們通過改造后端渲染方式,改為分段渲染。
const http = require("http");
const fs = require("fs");
var server = http.createServer((req, res)=>{
if(req.url!=="/favicon.ico"){
res.writeHead(200, {
'Content-Type' : 'text/html',
'Transfer-Encoding' : 'chunked'
});
getDataOne((data1) => {
res.write('<!DOCTYPE html><html><head><title>測(cè)試render</title></head><body>helloword<p>$data1</p>'.replace(/\$data1/g, data1));
getDataTwo((data2) => {
res.end('<p>$data2</p></body></html>'.replace(/\$data2/g, data2));
})
});
}
}).listen(3000, '127.0.0.1');
function getDataOne(fn1){
setTimeout(() => {
fn1('1111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111111');
}, 5000);
}
function getDataTwo(fn2){
setTimeout(() => {
fn2('22222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222222');
}, 5000);
}
通過設(shè)置http首部: Transfer-Encoding: chunked 即開啟了分段傳輸?shù)哪Х?。該編碼方式存在http1.1中,一般在服務(wù)器生成HTTP回應(yīng)是無法確定信息大小的,這時(shí)用Content-Length就無法事先寫入長(zhǎng)度,而需要實(shí)時(shí)生成消息長(zhǎng)度,則服務(wù)器一般采用Chunked編碼。
在進(jìn)行Chunked編碼傳輸時(shí),在回復(fù)消息的頭部有transfer-coding并定義為Chunked,表示將用Chunked編碼傳輸內(nèi)容。 下面我們看下修改后的效果:

雖然總體的頁面?zhèn)鬏敃r(shí)間并沒有變化,但是通過該方式,我們將響應(yīng)時(shí)間縮短了一半,減少了用戶等待的時(shí)間。在具體業(yè)務(wù)中,我們可以講用戶需要先看到的部分進(jìn)行提前輸出,將后端處理耗時(shí)較久的部分延遲輸出,這就是分段傳輸渲染的優(yōu)勢(shì)。 注意如果服務(wù)器是nginx,有可能由于緩沖區(qū)的設(shè)置導(dǎo)致分段渲染無效,需要調(diào)整緩沖區(qū)大小。
相關(guān)文章
Nodejs?Docker鏡像體積優(yōu)化實(shí)踐詳解
這篇文章主要為大家介紹了Nodejs?Docker鏡像體積優(yōu)化實(shí)踐示例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2023-07-07
淺談在koa2中實(shí)現(xiàn)頁面渲染的全局?jǐn)?shù)據(jù)
本篇文章主要介紹了淺談在koa2中實(shí)現(xiàn)頁面渲染的全局?jǐn)?shù)據(jù),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧2017-10-10
NodeJS?GRPC?多個(gè)?.proto?文件的處理步驟
本文教程詳細(xì)介紹了在NodeJS環(huán)境中如何使用gRPC框架處理多個(gè).proto文件,步驟包括安裝依賴、定義.proto文件、生成gRPC代碼、實(shí)現(xiàn)服務(wù)器和客戶端以及運(yùn)行,適用于開發(fā)者在構(gòu)建分布式應(yīng)用時(shí)進(jìn)行接口定義和服務(wù)實(shí)現(xiàn)2024-10-10
Node.js中使用事件發(fā)射器模式實(shí)現(xiàn)事件綁定詳解
這篇文章主要介紹了Node.js中使用事件發(fā)射器模式實(shí)現(xiàn)事件綁定詳解,本文一并講解了回調(diào)模式、發(fā)射器模式、事件類型等基礎(chǔ)知識(shí)做了補(bǔ)充,需要的朋友可以參考下2014-08-08
nodejs中用npm初始化來創(chuàng)建package.json的實(shí)例講解
今天小編就為大家分享一篇nodejs中用npm初始化來創(chuàng)建package.json的實(shí)例講解,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。一起跟隨小編過來看看吧2018-10-10

