Rumah > hujung hadapan web > tutorial js > Membina Aplikasi SSR React Sedia Pengeluaran

Membina Aplikasi SSR React Sedia Pengeluaran

Mary-Kate Olsen
Lepaskan: 2025-01-05 11:51:40
asal
188 orang telah melayarinya

Building Production-Ready SSR React Applications

Dalam dunia di mana setiap milisaat penting, pemaparan bahagian pelayan telah menjadi keupayaan penting untuk aplikasi bahagian hadapan.

Panduan ini akan membimbing anda melalui corak asas untuk membina SSR sedia pengeluaran dengan React. Anda akan mendapat pemahaman tentang prinsip di sebalik rangka kerja berasaskan React dengan SSR terbina dalam (seperti Next.js) dan belajar cara mencipta penyelesaian tersuai anda sendiri.

Kod yang disediakan adalah sedia pengeluaran, menampilkan proses binaan lengkap untuk kedua-dua bahagian klien dan pelayan, termasuk Dockerfile. Dalam pelaksanaan ini, Vite digunakan untuk membina klien dan kod SSR, tetapi anda boleh menggunakan alat lain pilihan anda. Vite juga memberikan hot-reload semasa mod pembangunan untuk pelanggan.

Jika anda berminat dengan versi persediaan ini tanpa Vite, sila hubungi kami.

Jadual Kandungan

  • Apa itu SSR
  • Mencipta Apl
    • Memulakan Vite
    • Mengemas kini Komponen Reaksi
    • Buat Pelayan
    • Mengkonfigurasi Binaan
  • Penghalaan
  • Pelabuh
  • Kesimpulan

Apa itu SSR

Perenderan sisi pelayan (SSR) ialah teknik dalam pembangunan web di mana pelayan menjana kandungan HTML halaman web sebelum menghantarnya ke penyemak imbas. Tidak seperti pemaparan sisi klien tradisional (CSR), di mana JavaScript membina kandungan pada peranti pengguna selepas memuatkan shell HTML kosong, SSR menyampaikan HTML yang dipaparkan sepenuhnya terus dari pelayan.

Faedah utama SSR:

  • SEO dipertingkatkan: Memandangkan perangkak enjin carian menerima kandungan yang dipaparkan sepenuhnya, SSR memastikan pengindeksan dan ranking.
  • Cat Pertama yang Lebih Cepat: Pengguna melihat kandungan yang bermakna dengan serta-merta, kerana pelayan mengendalikan pengangkatan berat rendering.
  • Prestasi Dipertingkat: Dengan mengurangkan beban kerja pemaparan pada penyemak imbas, SSR menyediakan pengalaman yang lebih lancar untuk pengguna pada peranti yang lebih lama atau kurang berkuasa.
  • Pemindahan Data Pelayan-ke-Pelanggan yang lancar: SSR membolehkan anda menghantar data bahagian pelayan dinamik kepada pelanggan tanpa membina semula berkas pelanggan.

Mencipta Apl

Aliran apl anda dengan SSR mengikut langkah berikut:

  1. Baca fail HTML templat.
  2. Mulakan React dan jana rentetan HTML kandungan apl.
  3. Suntikan rentetan HTML yang dijana ke dalam templat.
  4. Hantar HTML lengkap ke penyemak imbas.
  5. Pada klien, padankan teg HTML dan hidratkan aplikasi, menjadikannya interaktif.

Memulakan Vite

Saya lebih suka menggunakan templat pnpm dan react-swc-ts Vite, tetapi anda boleh memilih sebarang persediaan lain.

pnpm create vite react-ssr-app --template react-swc-ts
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk

Pasang kebergantungan:

pnpm create vite react-ssr-app --template react-swc-ts
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk

Mengemas kini Komponen React

Dalam aplikasi React biasa, terdapat satu titik masuk utama.tsx untuk index.html. Dengan SSR, anda memerlukan dua titik masuk: satu untuk pelayan dan satu untuk pelanggan.

Titik Masuk Pelayan

Pelayan Node.js akan menjalankan apl anda dan menjana HTML dengan menjadikan komponen React anda kepada rentetan (renderToString).

pnpm install
Salin selepas log masuk
Salin selepas log masuk

Titik Kemasukan Pelanggan

Pelayar akan menghidratkan HTML yang dijana pelayan, menyambungkannya dengan JavaScript untuk menjadikan halaman itu interaktif.

Penghidratan ialah proses melampirkan pendengar acara dan gelagat dinamik lain pada HTML statik yang diberikan oleh pelayan.

// ./src/entry-server.tsx
import { renderToString } from 'react-dom/server'
import App from './App'

export function render() {
  return renderToString(<App />)
}
Salin selepas log masuk
Salin selepas log masuk

Mengemas kini index.html

Kemas kini fail index.html dalam akar projek anda. pemegang tempat ialah tempat pelayan akan menyuntik HTML yang dijana.

// ./src/entry-client.tsx
import { hydrateRoot } from 'react-dom/client'
import { StrictMode } from 'react'
import App from './App'

import './index.css'

hydrateRoot(
  document.getElementById('root')!,
  <StrictMode>
    <App />
  </StrictMode>,
)
Salin selepas log masuk
Salin selepas log masuk

Semua kebergantungan yang diperlukan untuk pelayan hendaklah dipasang sebagai kebergantungan pembangunan (devDependencies) untuk memastikan kebergantungan itu tidak disertakan dalam himpunan klien.

Seterusnya, buat folder dalam akar projek anda bernama ./server dan tambahkan fail berikut.

Mengeksport semula Fail Pelayan Utama

Eksport semula fail pelayan utama. Ini menjadikan perintah menjalankan lebih mudah.

<!doctype html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <link rel="icon" type="image/svg+xml" href="/vite.svg" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>Vite + React + TS</title>
  </head>
  <body>
    <div>



<h3>
  
  
  Create Server
</h3>

<p>First, install the dependencies:<br>
</p>

<pre class="brush:php;toolbar:false">pnpm install -D express compression sirv tsup vite-node nodemon @types/express @types/compression
Salin selepas log masuk
Salin selepas log masuk

Mentakrifkan Pemalar

Pemalar HTML_KEY mesti sepadan dengan ulasan pemegang tempat dalam index.html. Pemalar lain mengurus tetapan persekitaran.

// ./server/index.ts
export * from './app'
Salin selepas log masuk
Salin selepas log masuk

Mencipta Pelayan Ekspres

Sediakan pelayan Express dengan konfigurasi berbeza untuk pembangunan dan persekitaran pengeluaran.

// ./server/constants.ts
export const NODE_ENV = process.env.NODE_ENV || 'development'
export const APP_PORT = process.env.APP_PORT || 3000

export const PROD = NODE_ENV === 'production'
export const HTML_KEY = `<!--app-html-->`
Salin selepas log masuk
Salin selepas log masuk

Konfigurasi Mod Pembangunan

Dalam pembangunan, gunakan perisian tengah Vite untuk mengendalikan permintaan dan mengubah fail index.html secara dinamik dengan muat semula panas. Pelayan akan memuatkan aplikasi React dan menjadikannya HTML pada setiap permintaan.

// ./server/app.ts
import express from 'express'
import { PROD, APP_PORT } from './constants'
import { setupProd } from './prod'
import { setupDev } from './dev'

export async function createServer() {
  const app = express()

  if (PROD) {
    await setupProd(app)
  } else {
    await setupDev(app)
  }

  app.listen(APP_PORT, () => {
    console.log(`http://localhost:${APP_PORT}`)
  })
}

createServer()
Salin selepas log masuk
Salin selepas log masuk

Konfigurasi Mod Pengeluaran

Dalam pengeluaran, gunakan pemampatan untuk mengoptimumkan prestasi, sirv untuk menyampaikan fail statik dan himpunan pelayan pra-bina untuk memaparkan apl.

// ./server/dev.ts
import { Application } from 'express'
import fs from 'fs'
import path from 'path'
import { HTML_KEY } from './constants'

const HTML_PATH = path.resolve(process.cwd(), 'index.html')
const ENTRY_SERVER_PATH = path.resolve(process.cwd(), 'src/entry-server.tsx')

export async function setupDev(app: Application) {
  // Create a Vite development server in middleware mode
  const vite = await (
    await import('vite')
  ).createServer({
    root: process.cwd(),
    server: { middlewareMode: true },
    appType: 'custom',
  })

  // Use Vite middleware for serving files
  app.use(vite.middlewares)

  app.get('*', async (req, res, next) => {
    try {
      // Read and transform the HTML file
      let html = fs.readFileSync(HTML_PATH, 'utf-8')
      html = await vite.transformIndexHtml(req.originalUrl, html)

      // Load the entry-server.tsx module and render the app
      const { render } = await vite.ssrLoadModule(ENTRY_SERVER_PATH)
      const appHtml = await render()

      // Replace the placeholder with the rendered HTML
      html = html.replace(HTML_KEY, appHtml)
      res.status(200).set({ 'Content-Type': 'text/html' }).end(html)
    } catch (e) {
      // Fix stack traces for Vite and handle errors
      vite.ssrFixStacktrace(e as Error)
      console.error((e as Error).stack)
      next(e)
    }
  })
}
Salin selepas log masuk
Salin selepas log masuk

Mengkonfigurasi Binaan

Untuk mengikuti amalan terbaik untuk membina aplikasi anda, anda harus mengecualikan semua pakej yang tidak diperlukan dan memasukkan hanya perkara yang sebenarnya digunakan oleh aplikasi anda.

Mengemas kini Konfigurasi Vite

Kemas kini konfigurasi Vite anda untuk mengoptimumkan proses binaan dan mengendalikan kebergantungan SSR:

// ./server/prod.ts
import { Application } from 'express'
import fs from 'fs'
import path from 'path'
import compression from 'compression'
import sirv from 'sirv'
import { HTML_KEY } from './constants'

const CLIENT_PATH = path.resolve(process.cwd(), 'dist/client')
const HTML_PATH = path.resolve(process.cwd(), 'dist/client/index.html')
const ENTRY_SERVER_PATH = path.resolve(process.cwd(), 'dist/ssr/entry-server.js')

export async function setupProd(app: Application) {
  // Use compression for responses
  app.use(compression())
  // Serve static files from the client build folder
  app.use(sirv(CLIENT_PATH, { extensions: [] }))

  app.get('*', async (_, res, next) => {
    try {
      // Read the pre-built HTML file
      let html = fs.readFileSync(HTML_PATH, 'utf-8')

      // Import the server-side render function and generate HTML
      const { render } = await import(ENTRY_SERVER_PATH)
      const appHtml = await render()

      // Replace the placeholder with the rendered HTML
      html = html.replace(HTML_KEY, appHtml)
      res.status(200).set({ 'Content-Type': 'text/html' }).end(html)
    } catch (e) {
      // Log errors and pass them to the error handler
      console.error((e as Error).stack)
      next(e)
    }
  })
}
Salin selepas log masuk
Salin selepas log masuk

Mengemas kini tsconfig.json

Kemas kini tsconfig.json anda untuk memasukkan fail pelayan dan mengkonfigurasi TypeScript dengan sewajarnya:

pnpm create vite react-ssr-app --template react-swc-ts
Salin selepas log masuk
Salin selepas log masuk
Salin selepas log masuk

Mencipta Konfigurasi tsup

Gunakan tsup, pengikat TypeScript, untuk membina kod pelayan. Pilihan noExternal menentukan pakej untuk digabungkan dengan pelayan. Pastikan anda memasukkan sebarang pakej tambahan yang digunakan oleh pelayan anda.

pnpm install
Salin selepas log masuk
Salin selepas log masuk

Menambah Skrip Binaan

// ./src/entry-server.tsx
import { renderToString } from 'react-dom/server'
import App from './App'

export function render() {
  return renderToString(<App />)
}
Salin selepas log masuk
Salin selepas log masuk

Menjalankan Aplikasi

Pembangunan: Gunakan arahan berikut untuk memulakan aplikasi dengan muat semula panas:

// ./src/entry-client.tsx
import { hydrateRoot } from 'react-dom/client'
import { StrictMode } from 'react'
import App from './App'

import './index.css'

hydrateRoot(
  document.getElementById('root')!,
  <StrictMode>
    <App />
  </StrictMode>,
)
Salin selepas log masuk
Salin selepas log masuk

Pengeluaran: Bina aplikasi dan mulakan pelayan pengeluaran:

<!doctype html>
<html lang="en">
  <head>
    <meta charset="UTF-8" />
    <link rel="icon" type="image/svg+xml" href="/vite.svg" />
    <meta name="viewport" content="width=device-width, initial-scale=1.0" />
    <title>Vite + React + TS</title>
  </head>
  <body>
    <div>



<h3>
  
  
  Create Server
</h3>

<p>First, install the dependencies:<br>
</p>

<pre class="brush:php;toolbar:false">pnpm install -D express compression sirv tsup vite-node nodemon @types/express @types/compression
Salin selepas log masuk
Salin selepas log masuk

Untuk mengesahkan bahawa SSR berfungsi, semak permintaan rangkaian pertama ke pelayan anda. Respons harus mengandungi HTML yang diberikan sepenuhnya bagi aplikasi anda.

Penghalaan

Untuk menambahkan halaman yang berbeza pada apl anda, anda perlu mengkonfigurasi penghalaan dengan betul dan mengendalikannya dalam kedua-dua titik masuk klien dan pelayan.

// ./server/index.ts
export * from './app'
Salin selepas log masuk
Salin selepas log masuk

Menambah Penghalaan Sebelah Pelanggan

Balut aplikasi anda dengan BrowserRouter dalam titik masuk klien untuk mendayakan penghalaan sisi klien.

// ./server/constants.ts
export const NODE_ENV = process.env.NODE_ENV || 'development'
export const APP_PORT = process.env.APP_PORT || 3000

export const PROD = NODE_ENV === 'production'
export const HTML_KEY = `<!--app-html-->`
Salin selepas log masuk
Salin selepas log masuk

Menambah Penghalaan Sisi Pelayan

Gunakan StaticRouter dalam titik masuk pelayan untuk mengendalikan penghalaan sebelah pelayan. Lulus url sebagai prop untuk memaparkan laluan yang betul berdasarkan permintaan.

// ./server/app.ts
import express from 'express'
import { PROD, APP_PORT } from './constants'
import { setupProd } from './prod'
import { setupDev } from './dev'

export async function createServer() {
  const app = express()

  if (PROD) {
    await setupProd(app)
  } else {
    await setupDev(app)
  }

  app.listen(APP_PORT, () => {
    console.log(`http://localhost:${APP_PORT}`)
  })
}

createServer()
Salin selepas log masuk
Salin selepas log masuk

Mengemas kini Konfigurasi Pelayan

Kemas kini kedua-dua persediaan pelayan pembangunan dan pengeluaran anda untuk menghantar URL permintaan kepada fungsi pemaparan:

// ./server/dev.ts
import { Application } from 'express'
import fs from 'fs'
import path from 'path'
import { HTML_KEY } from './constants'

const HTML_PATH = path.resolve(process.cwd(), 'index.html')
const ENTRY_SERVER_PATH = path.resolve(process.cwd(), 'src/entry-server.tsx')

export async function setupDev(app: Application) {
  // Create a Vite development server in middleware mode
  const vite = await (
    await import('vite')
  ).createServer({
    root: process.cwd(),
    server: { middlewareMode: true },
    appType: 'custom',
  })

  // Use Vite middleware for serving files
  app.use(vite.middlewares)

  app.get('*', async (req, res, next) => {
    try {
      // Read and transform the HTML file
      let html = fs.readFileSync(HTML_PATH, 'utf-8')
      html = await vite.transformIndexHtml(req.originalUrl, html)

      // Load the entry-server.tsx module and render the app
      const { render } = await vite.ssrLoadModule(ENTRY_SERVER_PATH)
      const appHtml = await render()

      // Replace the placeholder with the rendered HTML
      html = html.replace(HTML_KEY, appHtml)
      res.status(200).set({ 'Content-Type': 'text/html' }).end(html)
    } catch (e) {
      // Fix stack traces for Vite and handle errors
      vite.ssrFixStacktrace(e as Error)
      console.error((e as Error).stack)
      next(e)
    }
  })
}
Salin selepas log masuk
Salin selepas log masuk

Dengan perubahan ini, anda kini boleh membuat laluan dalam apl React anda yang serasi sepenuhnya dengan SSR. Walau bagaimanapun, pendekatan asas ini tidak mengendalikan komponen yang dimuatkan malas (React.lazy). Untuk menguruskan modul yang malas dimuatkan, sila rujuk artikel saya yang lain, Teknik SSR Reaksi Lanjutan dengan Penstriman dan Data Dinamik, dipautkan di bahagian bawah.

Docker

Berikut ialah Fail Docker untuk menyimpan aplikasi anda:

// ./server/prod.ts
import { Application } from 'express'
import fs from 'fs'
import path from 'path'
import compression from 'compression'
import sirv from 'sirv'
import { HTML_KEY } from './constants'

const CLIENT_PATH = path.resolve(process.cwd(), 'dist/client')
const HTML_PATH = path.resolve(process.cwd(), 'dist/client/index.html')
const ENTRY_SERVER_PATH = path.resolve(process.cwd(), 'dist/ssr/entry-server.js')

export async function setupProd(app: Application) {
  // Use compression for responses
  app.use(compression())
  // Serve static files from the client build folder
  app.use(sirv(CLIENT_PATH, { extensions: [] }))

  app.get('*', async (_, res, next) => {
    try {
      // Read the pre-built HTML file
      let html = fs.readFileSync(HTML_PATH, 'utf-8')

      // Import the server-side render function and generate HTML
      const { render } = await import(ENTRY_SERVER_PATH)
      const appHtml = await render()

      // Replace the placeholder with the rendered HTML
      html = html.replace(HTML_KEY, appHtml)
      res.status(200).set({ 'Content-Type': 'text/html' }).end(html)
    } catch (e) {
      // Log errors and pass them to the error handler
      console.error((e as Error).stack)
      next(e)
    }
  })
}
Salin selepas log masuk
Salin selepas log masuk

Membina dan Menjalankan Imej Docker

// ./vite.config.ts
import { defineConfig } from 'vite'
import react from '@vitejs/plugin-react-swc'
import { dependencies } from './package.json'

export default defineConfig(({ mode }) => ({
  plugins: [react()],
  ssr: {
    noExternal: mode === 'production' ? Object.keys(dependencies) : undefined,
  },
}))
Salin selepas log masuk
{
  "include": [
    "src",
    "server",
    "vite.config.ts"
  ]
}
Salin selepas log masuk

Kesimpulan

Dalam panduan ini, kami telah mewujudkan asas yang kukuh untuk mencipta aplikasi SSR sedia pengeluaran dengan React. Anda telah mempelajari cara menyediakan projek, mengkonfigurasi penghalaan dan mencipta Fail Docker. Persediaan ini sesuai untuk membina halaman pendaratan atau apl kecil dengan cekap.

Terokai Kod

  • Contoh: react-ssr-basics-example
  • Templat: templat-react-ssr
  • Templat Tambahan Vite: template-ssr-react-ts

Artikel Berkaitan

Ini adalah sebahagian daripada siri saya tentang SSR dengan React. Nantikan lebih banyak artikel!

  • Membina Aplikasi SSR React Sedia Pengeluaran (Anda di sini)
  • Teknik SSR React Terperinci dengan Penstriman dan Data Dinamik (Akan datang)
  • Menyediakan Tema dalam Aplikasi SSR React (Akan datang)

Kekal Terhubung

Saya sentiasa terbuka untuk maklum balas, kerjasama atau membincangkan idea teknologi — jangan ragu untuk menghubungi kami!

  • Portfolio: maxh1t.xyz
  • E-mel: m4xh17@gmail.com

Atas ialah kandungan terperinci Membina Aplikasi SSR React Sedia Pengeluaran. Untuk maklumat lanjut, sila ikut artikel berkaitan lain di laman web China PHP!

sumber:dev.to
Kenyataan Laman Web ini
Kandungan artikel ini disumbangkan secara sukarela oleh netizen, dan hak cipta adalah milik pengarang asal. Laman web ini tidak memikul tanggungjawab undang-undang yang sepadan. Jika anda menemui sebarang kandungan yang disyaki plagiarisme atau pelanggaran, sila hubungi admin@php.cn
Artikel terbaru oleh pengarang
Tutorial Popular
Lagi>
Muat turun terkini
Lagi>
kesan web
Kod sumber laman web
Bahan laman web
Templat hujung hadapan