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

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

實現(xiàn)ssr服務(wù)端渲染的方法步驟

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

前言

前段時間尋思做個個人網(wǎng)站,然后就立馬行動了。  個人網(wǎng)站如何實現(xiàn)選擇什么技術(shù)方案,自己可以自由決定。  剛好之前有大致想過服務(wù)端渲染,加載速度快,還有seo挺適合個人網(wǎng)站的。  所以就自己造了個輪子用koa+react來實現(xiàn)ssr服務(wù)端渲染。

什么是ssr

最初聽說有單頁面的服務(wù)端渲染的時候,就理解為類似傳統(tǒng)的服務(wù)端路由+模板渲染,只是需要用單頁面應(yīng)用的框架寫。后面尋思這樣好像有點(diǎn)傻,再一了解,原來只是在首次加載的時候,后端進(jìn)行當(dāng)前路徑頁面的組件渲染和數(shù)據(jù)請求,組裝成html返回給前端,用戶就能很快看到看到頁面,當(dāng)html中的js資源加載完成后,剩下執(zhí)行和運(yùn)行的就是一般的單頁面應(yīng)用。  所以ssr是后端模板渲染和單頁面的組合。  ssr有兩種模式,單頁面和非單頁面模式,第一種是后端首次渲染的單頁面應(yīng)用,第二種是完全使用后端路由的后端模版渲染模式。他們區(qū)別在于使用后端路由的程度。

優(yōu)勢

ssr的兩個明顯的優(yōu)勢:首次加載快和seo。  為什么說首次加載快呢。  一個普通的單頁面應(yīng)用,首次加載的時候需要把所有相關(guān)的靜態(tài)資源加載完畢,然后核心js才會開始執(zhí)行,這個過程就會消耗一定的時間,接著還會請求網(wǎng)絡(luò)接口,最終才能完全渲染完成。

ssr模式下,后端攔截到路由,找到對應(yīng)組件,準(zhǔn)備渲染組件,所有的js資源在本地,排除了js資源的網(wǎng)絡(luò)加載時間,接著只需要對當(dāng)前路由的組件進(jìn)行渲染,而頁面的ajax請求,可能在同一臺服務(wù)器上,如果是的話速度也會快很多。最后后端把渲染好的頁面反回給前端。  注意:頁面能很快的展示出來,但是由于當(dāng)前返回的只是單純展示的dom、css,其中的js相關(guān)的事件等在客戶端其實并沒有綁定,所以最終還是需要js加載完以后,對當(dāng)前的頁面再進(jìn)行一次渲染,稱為同構(gòu)。  所以ssr就是更快的先展示出頁面的內(nèi)容,先讓用戶能夠看到。  為什么seo友好呢,因為搜索引擎爬蟲在爬取頁面信息的時候,會發(fā)送HTTP請求來獲取網(wǎng)頁內(nèi)容,而我們服務(wù)端渲染首次的數(shù)據(jù)是后端返回的,返回的時候已經(jīng)是渲染好了title,內(nèi)容等信息,便于爬蟲抓取內(nèi)容。

如何實現(xiàn)

大致對ssr有了一個了解,我們現(xiàn)在需要對實現(xiàn)整理一下大致實現(xiàn)思路和流程。

1.選擇一個單頁面框架(我目前選擇的是react)

2.選擇node服務(wù)端框架(我目前選擇的是koa2)

3.實現(xiàn)核心邏輯,讓node服務(wù)端能夠路由和渲染單頁面組件(這一點(diǎn)分為很多小實現(xiàn)點(diǎn),后面說)

4.優(yōu)化開發(fā)和發(fā)布環(huán)境自動化構(gòu)建工具(webpack)

開始實現(xiàn)之前創(chuàng)建一個react-ssr項目,項目下創(chuàng)建client和server目錄用于寫客戶端和服務(wù)端代碼,webpack目錄用于weppack文件配置。

1.react應(yīng)用

安裝react依賴,在client中創(chuàng)建好一個基礎(chǔ)的react文件夾結(jié)構(gòu),并寫好一個可以運(yùn)行的有路由配置的應(yīng)用,client文件目錄如下:

ssr,服務(wù)端,渲染

2.server應(yīng)用

安裝koa和相關(guān)依賴,在server中創(chuàng)建好一個基礎(chǔ)的服務(wù)端文件夾結(jié)構(gòu),并寫好一個簡單的可運(yùn)行的后端應(yīng)用服務(wù)。server文件夾如下:

ssr,服務(wù)端,渲染

3.核心實現(xiàn)

因為有倉庫代碼就不對基礎(chǔ)代碼做解釋,現(xiàn)在我們有一個可以單獨(dú)運(yùn)行的react單頁面應(yīng)用和一個后端應(yīng)用,他們都有各自的路由。接下來我們做改造,實現(xiàn)ssr的單頁面模式(非單頁面模式僅僅是做部分調(diào)整,因此這里只講實現(xiàn)單頁面模式)。

核心實現(xiàn)分為以下幾步:

1) 后端攔截路由,根據(jù)路徑找到需要渲染的react頁面組件X

2)調(diào)用組件X初始化時需要請求的接口,同步獲取到數(shù)據(jù)后,使用react的renderToString方法對組件進(jìn)行渲染,使其渲染出節(jié)點(diǎn)字符串。

3)后端獲取基礎(chǔ)html文件,把渲染出的節(jié)點(diǎn)字符串插入到body之中,同時也可以操作其中的title,script等節(jié)點(diǎn)。返回完整的html給客戶端。

4)客戶端獲取后端返回的html,展示并加載其中的js,最后完成react同構(gòu)。

1)我們在客戶端寫react的時候,router常規(guī)的會定義一個數(shù)組,存放組件和對應(yīng)的path,然后注冊路由,如下:

ssr,服務(wù)端,渲染

上面說過,實現(xiàn)ssr就是實現(xiàn)單頁面應(yīng)用+首次服務(wù)端渲染,所以我們本身就是做的一個單頁面應(yīng)用。  現(xiàn)在實現(xiàn)了單頁面應(yīng)用,需要實現(xiàn)首次服務(wù)端渲染。  服務(wù)端的應(yīng)用啟動以后,接受到url請求,比如訪問 http://localhost:9999/ ,后端服務(wù)獲取到當(dāng)前的path為/,這個時候我們就希望后端找到配置path為‘/'的上圖的Index組件,對其進(jìn)行渲染。  我們在client的router文件夾中建立兩個js文件index和pages:

ssr,服務(wù)端,渲染

pages 里配置路由路徑和組件的映射,代碼大致如下,使其能被客戶端路由和服務(wù)端路由同時使用。

ssr,服務(wù)端,渲染

在server路由中代碼大致是這樣的,在服務(wù)端獲取到get請求以后,匹配路徑,如果路徑path是有映射頁面組件的,獲取到此組件并渲染,這就是我們的第一步:后端攔截路由,根據(jù)路徑找到需要渲染的react頁面組件。 

ssr,服務(wù)端,渲染

2)如上圖,匹配到組件以后,執(zhí)行了組件的getInitialProps方法(和nextjs的命名保持一致),此方法是一個封裝的靜態(tài)方法,主要用于獲取初始化所需要的ajax數(shù)據(jù),在服務(wù)端會同步獲取,而后通過ssrData參數(shù)傳入組件prorps并執(zhí)行組件渲染。  此方法在客戶端依然是異步請求。  這一步比較重要,為什么我們需要一個靜態(tài)方法,而不是直接把請求寫在willmount中呢。  因為在服務(wù)端使用renderToString渲染組件時,生命周期只會執(zhí)行到willmount之后的第一次render,在willmount內(nèi)部,請求是異步的,第一次render完成的時候,異步的數(shù)據(jù)都沒有獲取到,這個時候renderToString就已經(jīng)返回了。  那我們頁面的初始化數(shù)據(jù)就沒有了,返回的html不是我們所期望的。  因此定義了一個靜態(tài)方法,在組件實例化之前獲取到這個方法,同步執(zhí)行,數(shù)據(jù)獲取完成后,通過props把數(shù)據(jù)傳入給組件進(jìn)行渲染。  那么這個方法是如何實現(xiàn)的呢?  我們根據(jù)代碼截圖來看base.js:

ssr,服務(wù)端,渲染

首先在client的pages里新建一個base組件,base繼承React.Component,所有pages里的頁面組件都需要繼承這個base,base有一個靜態(tài)方法getInitialProps,此方法主要是返回組件初始化需要的異步數(shù)據(jù)。  如果有初始化的ajax請求,就應(yīng)該重寫在此方法里,并且return數(shù)據(jù)對象。 constructor判斷了頁面組件是否有初始化定義的state靜態(tài)方法,有的話傳遞給組件實例化的state對象,如果props有傳入ssrData,把ssrData傳遞值給組件state對象。   base中的componentWillMount會判斷是否還需要去執(zhí)行g(shù)etInitialProps方法,如果在服務(wù)端渲染的時候,數(shù)據(jù)已經(jīng)在組件實例化之前同步獲取并傳入了props,所以忽略。  如果在客戶端環(huán)境,分兩種情況,第一種:用戶第一次進(jìn)到頁面,這時候是服務(wù)端去請求的數(shù)據(jù),服務(wù)端獲取到數(shù)據(jù)后在服務(wù)端渲染組件,同時也會把數(shù)據(jù)存放在html的script代碼中,定義一個全局變量ssrData,如下圖,react在注冊單頁面應(yīng)用并且同構(gòu)的時候會把全局ssrData傳遞給頁面組件,這個時候頁面組件在客戶端同構(gòu)渲染的時候,就可以延續(xù)使用服務(wù)端之前的數(shù)據(jù),這樣也保持了同構(gòu)的一致性,也避免了一次重復(fù)請求。  第二種情況:就是當(dāng)前用戶在單頁面之中切換路由,這樣就沒有服務(wù)端渲染,那么就執(zhí)行g(shù)etInitialProps方法,把數(shù)據(jù)直接返回給state,幾乎等同于在willmount中執(zhí)行請求。  這樣封裝我們就可以用一套代碼兼容服務(wù)端渲染和單頁面渲染。 

ssr,服務(wù)端,渲染

client/app.js

ssr,服務(wù)端,渲染

再看看如何寫頁面組件,下面是頁面組件Index的截圖,Index繼承Base,定義了靜態(tài)state,組件constructor方法會把此對象傳遞給組件實例化的state對象中,之所以用靜態(tài)方法來寫默認(rèn)數(shù)據(jù),是想保證定義的默認(rèn)state先傳遞給實例對象的state,接口請求傳遞的props數(shù)據(jù)后傳遞給實例對象的state。 為什么不直接寫state屬性而要加static,因為state屬性會執(zhí)行在constructor之后,這樣會覆蓋constructor定義的state,也就是會覆蓋我們getInitialProps返回的數(shù)據(jù)。

ssr,服務(wù)端,渲染

注意:在服務(wù)端渲染環(huán)境下,執(zhí)行renderToString的時候,組件會被實例化,并且返回字符串形式的dom,這個過程react組件的生命周期只會執(zhí)行到willmount之后的render。

3)我們寫好一個html文件,大致如下。  當(dāng)前已經(jīng)渲染出了相應(yīng)的節(jié)點(diǎn)字符串,后端需要返回html文本,內(nèi)容應(yīng)該包含標(biāo)題,節(jié)點(diǎn)和最后需要加載的打包好的js,依次去替換html占位部分。 

index.html

ssr,服務(wù)端,渲染

server/router.js

ssr,服務(wù)端,渲染

4)最后客戶端js加載完成后,會運(yùn)行react,并且執(zhí)行同構(gòu)方法ReactDOM.hydrate,而不是平時用的ReactDOM.render。

ssr,服務(wù)端,渲染

以下是首次渲染過程大致流程圖,點(diǎn)擊查看大圖

ssr,服務(wù)端,渲染

css處理

現(xiàn)在我們已經(jīng)完成了最核心的邏輯,但是有一個問題。  我發(fā)現(xiàn)在后端渲染組件的時候,style-loader會報錯,style-loader會找到組件依賴的css,并在組件加載時,把style載入到html header中,但是我們在服務(wù)端渲染的時候,沒有window對象,因此style-loader內(nèi)部代碼會報錯。    服務(wù)端webpack需要移除style-loader,用其他方法代替,后來我把樣式賦值給組件靜態(tài)變量,然后通過服務(wù)端渲染一并返回給前端,但是有個問題,我只能拿到當(dāng)前組件的樣式,子組件的樣式?jīng)]辦法拿到,如果要給子組件再添加靜態(tài)方法,再想辦法去取,那就太麻煩了。  后來我找到了一個庫isomorphic-style-loader可以支持我們想要的功能,看了下它的源碼和使用方法,通過高階函數(shù)把樣式賦值給組件,然后利用react的Context,拿到當(dāng)前需要渲染的所有組件的樣式,最后把style插入到html中,這樣解決了子組件樣式無法導(dǎo)入的問題。  但是我覺得有點(diǎn)麻煩,首先需要定義所有組件的高階函數(shù)和引入這個庫,然后在router之中需要寫相關(guān)代碼收集style,最后插入到html中。  后來我定義了一個ProcessSsrStyle方法,入?yún)⑹莝tyle文件,邏輯是判斷環(huán)境,如果是服務(wù)端把style加載到當(dāng)前組件的dom中,如果是客戶端就不處理(因為客戶端有style-loader)。  實現(xiàn)和使用非常簡單,如下:

ProcessSsrStyle.js

ssr,服務(wù)端,渲染

使用:

ssr,服務(wù)端,渲染

服務(wù)端返回html的內(nèi)容如下,用戶馬上能夠看到完整的頁面樣式,而當(dāng)客戶端react同構(gòu)完成后,dom會被替換為純dom,因為ProcessSsrStyle方法在客戶端不會輸出style,最終style-loader執(zhí)行后header中也會有樣式,,頁面不會出現(xiàn)不一致的變化,對于用戶來說這一切都是無感的。

ssr,服務(wù)端,渲染

至此,最核心的功能已經(jīng)實現(xiàn),但是在后來的開發(fā)中,我發(fā)現(xiàn)事情還并沒有那么簡單,因為開發(fā)環(huán)境似乎太不友好了,開發(fā)效率低,需要手動重啟。 

開發(fā)環(huán)境

先說說最初的開發(fā)環(huán)境如何工作:

  • npm run dev啟動開發(fā)環(huán)境
  • webpack.client-dev.js打包服務(wù)端代碼,代碼會被打包到dist/server中
  • webpack.server-dev.js打包客戶端代碼,代碼會被打包到dist/client中
  • 啟動服務(wù)端應(yīng)用,端口9999
  • 啟動webpack-dev-server, 端口8888

webpack打包后,啟動了兩個服務(wù),一個是服務(wù)端的app應(yīng)用、端口為9999,一個是客戶端的dev-server、端口為8888,dev-server會監(jiān)聽和打包c(diǎn)lient代碼,可以在客戶端代碼更新的時候,實時熱更新前端代碼。  當(dāng)訪問localhost:9999時,server會返回html,我們的server返回的html中的js腳本路徑是指向的dev-serve端口的地址,如下圖。  也就是說,客戶端的程序和服務(wù)端的程序被分別打包,并且運(yùn)行兩個不同的端口服務(wù)。  

ssr,服務(wù)端,渲染

在生產(chǎn)環(huán)境下,因為不需要dev-server去監(jiān)聽和熱更新,因此只一個服務(wù)就足夠, 如下圖,服務(wù)端注冊靜態(tài)資源文件夾:

server/app.js

ssr,服務(wù)端,渲染

目前的構(gòu)建系統(tǒng),區(qū)分了生產(chǎn)環(huán)境和開發(fā)環(huán)境,現(xiàn)在的開發(fā)環(huán)境構(gòu)建是沒有什么問題的。  但是開發(fā)環(huán)境問題就比較明顯,存在的最大問題是服務(wù)端沒有熱更新或者重新打包重啟。  這樣會導(dǎo)致很多問題,最嚴(yán)重的就是前端已經(jīng)更新了組件,但是服務(wù)端并沒有更新,所以在同構(gòu)的時候會出現(xiàn)不一致,就會導(dǎo)致報錯,有些報錯會影響運(yùn)行,解決辦法只有重啟。  這樣的開發(fā)體驗是無法忍受的。  后來我開始考慮做服務(wù)端的熱更新。

監(jiān)聽、打包、重啟

最初我的方法是監(jiān)聽修改,打包然后重啟應(yīng)用。  還記得我們的client/router/pages.js文件嗎,客戶端和服務(wù)端的路由都引入了這個文件,所以服務(wù)端和客戶端的打包依賴都有pages.js,因此所有pages的組件相關(guān)的依賴都可以被客戶端和服務(wù)端監(jiān)聽,當(dāng)一個組件更新了,dev-server已經(jīng)幫助我們監(jiān)聽和熱更新了客戶端代碼,現(xiàn)在我們要自己來處理以下如何更新和重啟服務(wù)端代碼。  其實方法很簡單,就是在服務(wù)端打包配置里開啟監(jiān)聽,然后在插件配置中,寫一個重啟的插件,插件代碼如下:

ssr,服務(wù)端,渲染

當(dāng)webpack首次運(yùn)行之后,插件會啟動一個子進(jìn)程,運(yùn)行app.js,當(dāng)文件發(fā)生變動后,再次編譯,判斷是否有子進(jìn)程,如果有殺掉子進(jìn)程,然后重啟子進(jìn)程,這樣就實現(xiàn)了自動重啟。  因為客戶端和服務(wù)端是兩個不同的打包服務(wù)和配置,當(dāng)文件被修改,他們同時會重新編譯,為了保證編譯后運(yùn)行符合預(yù)期,要保證服務(wù)端先編譯完成,客戶端后編譯完成,所以在客戶端的watch配置里,增加一點(diǎn)延遲,如下圖,默認(rèn)是300毫秒,所以服務(wù)端是300毫秒后執(zhí)行編譯,而客戶端是1000毫秒后執(zhí)行編譯。

ssr,服務(wù)端,渲染

現(xiàn)在解決了重啟問題,但是我覺得還不夠,因為在開發(fā)的大部分時間里pages.js中組件,也就是展示端的代碼更新頻率會很高,如果老是去重啟編譯后端的代碼,我覺得效率太低。  因此我覺得再做一次優(yōu)化。

抽離client/router/pages單獨(dú)打包

流程應(yīng)該是這樣的,增加一個webpack.server-dev-pages.js配置文件,單獨(dú)監(jiān)聽和打包出dist/pages,服務(wù)端代碼判斷如果是開發(fā)環(huán)境,在路由監(jiān)聽方法中每次執(zhí)行都重新獲取dist/pages包,服務(wù)端監(jiān)聽配置忽略client文件夾。  看起來有點(diǎn)懵逼,其實最終的效果就是當(dāng)pages中依賴的組件發(fā)生了更新,webpack.server-dev-pages.js重新編譯并打包到dist/pages中,服務(wù)端app不編譯和重啟,只需要在服務(wù)端app路由中重新獲取最新的dist/pages包,就保證了服務(wù)應(yīng)用更新了所有客戶端組件,而服務(wù)端應(yīng)用并不會編譯和重啟。  當(dāng)服務(wù)端本身的代碼發(fā)生了修改,還是會自動編譯和重啟。  所以最終我們的開發(fā)環(huán)境需要啟動3個打包配置

  • webpack.server-dev-pages
  • webpack.server-dev
  • webpack.client-dev

server/router,如何清除和更新pages包

ssr,服務(wù)端,渲染

至此,比較滿意的開發(fā)環(huán)境基本實現(xiàn)了。  后來又覺得每次更新css都需要去重新打包后端的pages也沒有必要,加上同構(gòu)的時候css不一致,僅僅只有警告,沒有實質(zhì)影響,因此我在server-dev-pages中忽略了less文件(因為我用的less)。  這樣會導(dǎo)致一個問題,因為沒有更新pages,所以頁面會刷新時會先展示舊的樣式,然后同構(gòu)完成又立馬變成新樣式,在開發(fā)環(huán)境中這一瞬間是可以接受的,也不影響什么。  但是避免了無謂的編譯。

ssr,服務(wù)端,渲染

沒有做的事情

  • 封裝成一個更有包裹性的三方腳手架
  • css作用域控制
  • 封裝性更強(qiáng)的webpack配置
  • 開發(fā)環(huán)境下,圖片路徑會出現(xiàn)不一致

最初做自己小站的目的是學(xué)習(xí),加上自己使用,因此有太多個性的東西。  從自己的小站中抽離了出來,已經(jīng)刪去了很多包和代碼,只為了讓他人更能快速理解其中的核心代碼。  代碼中有很多注釋都能幫助他人理解,如果大家想使用當(dāng)前庫開發(fā)一個自己的小站,是完全可以的,也可以幫助大家更好的理解它。  如果是用于商業(yè)項目,推薦nextjs。  css沒有做作用域控制,因此如果想隔離作用域,手動添加上層css隔離,比如.index{ ..... }包裹一層,或者嘗試自己引入三方包。  webpack通用的配置可以封裝成一個文件,然后在每個文件里引入,再個性修改。  但是之前看其他代碼的時候發(fā)現(xiàn),這種方法,會增加閱讀難度,加上本身配置內(nèi)容不多,所以不做封裝,看起來更直觀。  開發(fā)環(huán)境下,圖片路徑會出現(xiàn)不一致,比如客戶端地址請求地址是localhost...assets/xx.jpg,而服務(wù)端是assets/xx.jpg,可能會有警告,但是不影響。  因為只是一個是絕對路徑,一個是相對路徑。 

ssr,服務(wù)端,渲染

最后

對于這次的ssr服務(wù)端渲染的實現(xiàn)還是挺滿意的,也花費(fèi)了挺多時間。  感受下加載速度吧,歡迎訪問大詩人小站,https://dashiren.cn/ 。  部分頁面有接口請求,比如https://dashiren.cn/space,加載速度依然很快。

ssr,服務(wù)端,渲染

倉庫已經(jīng)準(zhǔn)備好,下載下來試試吧,安裝依賴后,運(yùn)行命令即可。https://github.com/zimv/react-ssr

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


注:相關(guān)教程知識閱讀請移步到JavaScript/Ajax教程頻道。
發(fā)表評論 共有條評論
用戶名: 密碼:
驗證碼: 匿名發(fā)表
主站蜘蛛池模板: 柳林县| 建宁县| 安丘市| 德州市| 宿迁市| 洪洞县| 镶黄旗| 武陟县| 山东省| 合水县| 霍城县| 合水县| 泰州市| 广宗县| 日土县| 垫江县| 石楼县| 肇源县| 专栏| 漳州市| 改则县| 西昌市| 扎囊县| 洛扎县| 个旧市| 健康| 仁寿县| 岳池县| 济阳县| 文水县| 渝中区| 南宁市| 景德镇市| 达拉特旗| 聂拉木县| 三亚市| 平江县| 广宁县| 文山县| 澄迈县| 北票市|