使用React?SSR寫Demo一學就會
引言
今天寫個小 Demo
來從頭實現(xiàn)一下 react
的 SSR
,幫助理解 SSR
是如何實現(xiàn)的,有什么細節(jié)。
什么是 SSR
SSR
即 Server Side Rendering
服務端渲染,是指將網(wǎng)頁內(nèi)容在服務器端中生成并發(fā)送到瀏覽器的技術。相比于客戶端渲染(CSR
),SSR
一般用于以下場景:
SEO
(搜索引擎優(yōu)化):由于部分搜索引擎對CSR
內(nèi)容支持不佳,所以SSR
可以提升網(wǎng)站在搜索引擎結(jié)果中的排名。- 首屏加載速度:由于
SSR
可以在服務器端生成完整的HTML
頁面,用戶打開網(wǎng)頁時能夠更快地看到內(nèi)容,不會看到長時間的白屏,可以提升用戶體驗。 - 隱藏某些數(shù)據(jù):由于
CSR
需要從服務器將數(shù)據(jù)下載下來進行動態(tài)渲染,所以一些數(shù)據(jù)很容易被他人獲取,而SSR
由于數(shù)據(jù)到渲染的過程在服務端實現(xiàn),所以可以用來隱藏一些不想讓他人輕易獲得的數(shù)據(jù)。
如何實現(xiàn)
簡單的 SSR
其實實現(xiàn)很簡單,只需要在服務端導入要渲染的組件,然后調(diào)用 react-dom/server
包中提供的 renderToString
方法將該組件的渲染內(nèi)容輸出為字符串后返回客戶端即可。
Server 端的組件
下面寫一個簡單的例子:
服務端代碼:
import express from 'express'; import React from 'react'; import { renderToString } from 'react-dom/server'; import App from '../ui/App'; const app = express(); app.get('/', (_: unknown, res: express.Response) => { res.send(renderToString(<App />)); }); app.listen(4000, () => { console.log('Listening on port 4000'); });
此處要注意服務端需要支持 jsx
語法的解析,我這里直接使用 esno
執(zhí)行 ts
代碼,在 tsconfig.json
中配置 jsx
即可。
其實看到這里就能明白為什么在 SSR
的頁面上使用 window
、localstorage
等瀏覽器 API
需要放到 useEffect
里了,因為該頁面的組件都會被 server
端讀取解析,而 server
端并沒有這些 API
。
然后看下 App
組件的代碼:
import React, { useCallback } from 'react'; export default () => { const log = useCallback(() => { console.log('Hello world'); }, []); return ( <div> <p>react ssr demo</p> <button onClick={log}>Click me</button> </div> ); };
啟動服務器后 server
端就會使用 renderToString
將 <App />
渲染成 html
字符串,然后通過 send
返回給前端,下面就是服務端返回的 html
內(nèi)容:
<div> <p>react ssr demo</p> <button>Click me</button> </div>
打開瀏覽器訪問該地址即可看到服務端返回了該 html
片段:
hydrate 復活組件
如果你跟著上面的操作很快就會發(fā)現(xiàn)問題:為什么點按鈕沒法操作了?
其實原因很簡單,因為我們只拿到了一個 html
并沒有任何的 js
,事件綁定等自然是無法實現(xiàn)的,要復活組件的交互我們還需要很重要的一步 - hydrate
也就是常說的水合。
hydrate
即通過 react
將對應的組件重新渲染到 SSR
渲染的靜態(tài)內(nèi)容上,類似于 render
差異點在于 render
會忽略 root
元素中現(xiàn)有的 dom
而 hydrate
則會復用并會進行內(nèi)容匹配檢查。
Hydration failed because the initial UI does not match what was rendered on the server.
如果遇到上述錯誤即表示在客戶端執(zhí)行 hydrate
時服務端返回的初始的 dom
和 hydrate
接收到的需要進行渲染的 dom
不匹配。
說了這么多我們再來看下代碼如何編寫,首先要進行 hydrate
我們需要客戶端的代碼來執(zhí)行:
import React from 'react'; import { hydrateRoot } from 'react-dom/client'; import App from './App'; hydrateRoot(document.getElementById('root')!, <App />);
然后將該代碼進行編譯打包,我這里就直接使用 webpack
進行打包:
const path = require('path'); module.exports = { entry: './ui/index.tsx', output: { path: path.resolve(__dirname, 'static'), filename: 'bundle.js' }, resolve: { extensions: ['.ts', '.tsx', '.js', '.jsx'] }, module: { rules: [ { test: /\.(t|j)sx?$/, exclude: /node_modules/, use: { loader: 'babel-loader', options: { presets: ['@babel/preset-react', '@babel/preset-typescript'] } } } ] } };
打包完成后生成一個 bundle.js
即可在客戶端使用它來進行 hydrate
。
然后我們再修改下 server
端的代碼:
app.get('/', (_: unknown, res: express.Response) => { res.send( ` <div id="root">${renderToString(<App />)}</div> <script src="/bundle.js"></script> ` ); }); app.use(express.static('static'));
我們在靜態(tài)內(nèi)容的外層套上 root
元素,然后在下方引入我們剛剛編譯的腳本,然后就可以在客戶端看到我們想要的結(jié)果:
可以看到事件可以正常觸發(fā)了。
此處還有個注意點,在 server
端要注意將靜態(tài)字符串包裹在 root
元素中不要添加換行空格等,不然 react
在 hydrate
時依舊會因為內(nèi)容不匹配而提示 Hydration failed
(僅在 hydrateRoot
時出現(xiàn),如果使用 hydrate
不會報錯,不過 18 中 hydrate
已經(jīng)被棄用。)
動態(tài)數(shù)據(jù)
此時有些同學可能發(fā)現(xiàn)一些問題:前面的內(nèi)容所渲染的內(nèi)容都是靜態(tài)的,如果要針對用戶渲染出不同的內(nèi)容比如用戶信息等如何是好?
其實很簡單,只需要在服務端將對應的信息作為 props
進行渲染即可,我們下面使用 userName
模擬一下:
app.get('/', (_: unknown, res: express.Response) => { const userName = ['張三', '李四', '王五', '趙六'][(Math.random() * 4) | 0]; res.send( ` <div id="root">${renderToString(<App userName={userName} />)}</div> <script src="/bundle.js"></script> ` ); });
可是客戶端要如何與服務端匹配呢?此處有兩種解決方案:
- 客戶端獲取對應的信息并在信息獲取完成后再進行
hydrate
操作。 - 服務端將獲取到的信息放在頁面中。
可以看出方案 1 會帶來明顯的延時,所以一般會采用方案 2,實現(xiàn)一般可以使用全局變量或特定標簽來實現(xiàn):
app.get('/', (_: unknown, res: express.Response) => { const userName = ['張三', '李四', '王五', '趙六'][(Math.random() * 4) | 0]; res.send( ` <div id="root">${renderToString(<App userName={userName} />)}</div> <script> window.__initialState = { userName: '${userName}' }; </script> <script src="/bundle.js"></script> ` ); });
import React from 'react'; import { hydrateRoot } from 'react-dom/client'; import App from './App'; hydrateRoot(document.getElementById('root')!, <App {...window.__initialState} />);
總結(jié)
React
中的SSR
可以通過renderToString
來實現(xiàn),但是只能輸出靜態(tài)內(nèi)容,要讓頁面支持交互需要搭配hydrate
使用。- 實現(xiàn)
SSR
時服務端需要支持jsx
語法的解析,因為服務端也需要讀取組件。 hydrate
會檢查服務端與客戶端的內(nèi)容是否匹配。- 要實現(xiàn)動態(tài)數(shù)據(jù)需要在客戶端與服務端之間做好如何使用初始
props
的約定。
最后
本文的 demo
代碼放置在 React SSR Demo 中,可自行取閱。
以上就是React SSR - 寫個 Demo 一學就會的詳細內(nèi)容,更多關于React SSR - 寫個 Demo 一學就會的資料請關注腳本之家其它相關文章!
相關文章
react antd-mobile ActionSheet+tag實現(xiàn)多選方式
這篇文章主要介紹了react antd-mobile ActionSheet+tag實現(xiàn)多選方式,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教2023-10-10React實現(xiàn)類似淘寶tab居中切換效果的示例代碼
這篇文章主要介紹了React實現(xiàn)類似淘寶tab居中切換效果,本文通過實例代碼給大家介紹的非常詳細,對大家的學習或工作具有一定的參考借鑒價值,需要的朋友可以參考下2020-06-06詳解React如何優(yōu)雅地根據(jù)prop更新state值
這篇文章主要為大家詳細介紹了React如何優(yōu)雅地實現(xiàn)根據(jù)prop更新state值,文中的示例代碼講解詳細,具有一定的參考價值,感興趣的小伙伴可以了解下2023-11-11React Native之prop-types進行屬性確認詳解
本篇文章主要介紹了React Native之prop-types進行屬性確認詳解,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-12-12React競態(tài)條件Race Condition實例詳解
這篇文章主要為大家介紹了React競態(tài)條件Race Condition實例詳解,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪2022-11-11React+TypeScript項目中使用CodeMirror的步驟
CodeMirror被廣泛應用于許多Web應用程序和開發(fā)工具,之前做需求用到過codeMirror這個工具,覺得還不錯,功能很強大,所以記錄一下改工具的基礎用法,對React+TypeScript項目中使用CodeMirror的步驟感興趣的朋友跟隨小編一起看看吧2023-07-07