Skip to content

构建高性能 SSR 应用

服务端渲染(SSR)在现代 Web 开发中扮演着重要角色,它能够显著改善首屏加载时间和 SEO 表现。然而,要构建一个真正高性能的 SSR 应用,需要在多个层面进行优化。本文将深入探讨 SSR 应用的性能优化策略和最佳实践。

目录

SSR 基础概念回顾

什么是服务端渲染?

服务端渲染是指在服务器上生成完整的 HTML 页面,然后发送给客户端的技术。与客户端渲染(CSR)相比,SSR 具有以下优势:

  • 更快的首屏加载:用户可以立即看到内容
  • 更好的 SEO:搜索引擎可以直接抓取完整的 HTML
  • 更好的社交媒体分享:Open Graph 标签可以被正确解析

SSR vs CSR vs SSG 对比

特性SSRCSRSSG
首屏加载最快
SEO 友好优秀优秀
服务器负载
动态内容支持支持有限
开发复杂度

性能优化策略

1. 缓存策略

缓存是 SSR 性能优化的核心。合理的缓存策略可以显著减少服务器负载和响应时间。

页面级缓存

js
// Express + Redis 页面缓存示例
const redis = require('redis');
const client = redis.createClient();

const pageCache = (duration = 300) => {
  return async (req, res, next) => {
    const key = `page:${req.originalUrl}`;
    
    try {
      const cached = await client.get(key);
      if (cached) {
        return res.send(cached);
      }
      
      // 拦截 res.send 来缓存响应
      const originalSend = res.send;
      res.send = function(body) {
        client.setex(key, duration, body);
        originalSend.call(this, body);
      };
      
      next();
    } catch (error) {
      console.error('缓存错误:', error);
      next();
    }
  };
};

// 使用页面缓存
app.get('/products/:id', pageCache(600), async (req, res) => {
  const product = await getProduct(req.params.id);
  const html = await renderToString(<ProductPage product={product} />);
  res.send(html);
});

组件级缓存

js
// Vue SSR 组件缓存
const LRU = require('lru-cache');

const microCache = new LRU({
  max: 100,
  ttl: 1000 * 60 * 15 // 15分钟
});

// 在 Vue 组件中使用缓存
const ProductCard = {
  name: 'ProductCard',
  serverCacheKey: props => `product-card:${props.id}`,
  props: ['id', 'name', 'price'],
  template: `
    <div class="product-card">
      <h3>{{ name }}</h3>
      <p>¥{{ price }}</p>
    </div>
  `
};

数据层缓存

js
// 数据获取缓存
class DataCache {
  constructor() {
    this.cache = new Map();
    this.ttl = new Map();
  }
  
  async get(key, fetcher, duration = 300000) {
    const now = Date.now();
    
    // 检查缓存是否过期
    if (this.cache.has(key) && this.ttl.get(key) > now) {
      return this.cache.get(key);
    }
    
    // 获取新数据
    const data = await fetcher();
    this.cache.set(key, data);
    this.ttl.set(key, now + duration);
    
    return data;
  }
  
  invalidate(pattern) {
    for (const key of this.cache.keys()) {
      if (key.includes(pattern)) {
        this.cache.delete(key);
        this.ttl.delete(key);
      }
    }
  }
}

const dataCache = new DataCache();

// 使用数据缓存
const getProductList = async (category) => {
  return dataCache.get(
    `products:${category}`,
    () => db.products.findMany({ where: { category } }),
    600000 // 10分钟缓存
  );
};

2. 代码分割与懒加载

代码分割可以减少初始包大小,提高加载速度。

路由级代码分割

js
// Next.js 动态导入
import dynamic from 'next/dynamic';

const DynamicComponent = dynamic(() => import('../components/HeavyComponent'), {
  loading: () => <p>加载中...</p>,
  ssr: false // 禁用 SSR,仅客户端渲染
});

// Vue Router 懒加载
const routes = [
  {
    path: '/dashboard',
    component: () => import('./views/Dashboard.vue')
  },
  {
    path: '/analytics',
    component: () => import('./views/Analytics.vue')
  }
];

组件级代码分割

js
// React 组件懒加载
import { lazy, Suspense } from 'react';

const LazyChart = lazy(() => import('./Chart'));

function Dashboard() {
  return (
    <div>
      <h1>仪表板</h1>
      <Suspense fallback={<div>图表加载中...</div>}>
        <LazyChart />
      </Suspense>
    </div>
  );
}

第三方库分割

js
// webpack 配置
module.exports = {
  optimization: {
    splitChunks: {
      chunks: 'all',
      cacheGroups: {
        vendor: {
          test: /[\\/]node_modules[\\/]/,
          name: 'vendors',
          chunks: 'all',
        },
        common: {
          name: 'common',
          minChunks: 2,
          chunks: 'all',
          enforce: true
        }
      }
    }
  }
};

3. 预渲染与静态生成

对于相对静态的内容,可以使用预渲染技术进一步提升性能。

增量静态再生(ISR)

js
// Next.js ISR 示例
export async function getStaticProps({ params }) {
  const product = await getProduct(params.id);
  
  return {
    props: {
      product,
    },
    // 每60秒重新验证页面
    revalidate: 60,
  };
}

export async function getStaticPaths() {
  // 预生成热门产品页面
  const popularProducts = await getPopularProducts();
  const paths = popularProducts.map((product) => ({
    params: { id: product.id.toString() },
  }));

  return {
    paths,
    // 其他页面按需生成
    fallback: 'blocking',
  };
}

自定义预渲染系统

js
// 预渲染脚本
const puppeteer = require('puppeteer');
const fs = require('fs').promises;
const path = require('path');

class PreRenderer {
  constructor(options = {}) {
    this.baseUrl = options.baseUrl || 'http://localhost:3000';
    this.outputDir = options.outputDir || './dist';
    this.routes = options.routes || [];
  }
  
  async render() {
    const browser = await puppeteer.launch();
    
    for (const route of this.routes) {
      await this.renderRoute(browser, route);
    }
    
    await browser.close();
  }
  
  async renderRoute(browser, route) {
    const page = await browser.newPage();
    
    try {
      await page.goto(`${this.baseUrl}${route}`, {
        waitUntil: 'networkidle0'
      });
      
      const html = await page.content();
      const filePath = path.join(this.outputDir, `${route}.html`);
      
      await fs.mkdir(path.dirname(filePath), { recursive: true });
      await fs.writeFile(filePath, html);
      
      console.log(`预渲染完成: ${route}`);
    } catch (error) {
      console.error(`预渲染失败 ${route}:`, error);
    } finally {
      await page.close();
    }
  }
}

// 使用预渲染
const preRenderer = new PreRenderer({
  baseUrl: 'http://localhost:3000',
  outputDir: './dist',
  routes: ['/', '/about', '/products', '/contact']
});

preRenderer.render();

4. 资源优化

图片优化

js
// Next.js Image 组件优化
import Image from 'next/image';

function ProductImage({ src, alt, ...props }) {
  return (
    <Image
      src={src}
      alt={alt}
      width={400}
      height={300}
      placeholder="blur"
      blurDataURL="..."
      sizes="(max-width: 768px) 100vw, (max-width: 1200px) 50vw, 33vw"
      {...props}
    />
  );
}

CSS 优化

js
// 关键 CSS 内联
const criticalCSS = `
  body { margin: 0; font-family: Arial, sans-serif; }
  .header { background: #333; color: white; padding: 1rem; }
  .main { max-width: 1200px; margin: 0 auto; padding: 2rem; }
`;

// 在 HTML 头部内联关键 CSS
const htmlTemplate = `
<!DOCTYPE html>
<html>
<head>
  <style>${criticalCSS}</style>
  <link rel="preload" href="/styles/main.css" as="style" onload="this.onload=null;this.rel='stylesheet'">
</head>
<body>
  ${appHtml}
</body>
</html>
`;

字体优化

html
<!-- 字体预加载 -->
<link rel="preload" href="/fonts/main.woff2" as="font" type="font/woff2" crossorigin>

<!-- 字体显示策略 -->
<style>
@font-face {
  font-family: 'CustomFont';
  src: url('/fonts/main.woff2') format('woff2');
  font-display: swap; /* 使用系统字体直到自定义字体加载完成 */
}
</style>

5. 服务器优化

Node.js 性能优化

js
// 集群模式
const cluster = require('cluster');
const numCPUs = require('os').cpus().length;

if (cluster.isMaster) {
  console.log(`主进程 ${process.pid} 正在运行`);
  
  // 衍生工作进程
  for (let i = 0; i < numCPUs; i++) {
    cluster.fork();
  }
  
  cluster.on('exit', (worker, code, signal) => {
    console.log(`工作进程 ${worker.process.pid} 已退出`);
    cluster.fork(); // 重启工作进程
  });
} else {
  // 工作进程运行应用
  require('./app.js');
  console.log(`工作进程 ${process.pid} 已启动`);
}

内存管理

js
// 内存监控
const monitorMemory = () => {
  const used = process.memoryUsage();
  
  console.log('内存使用情况:');
  for (let key in used) {
    console.log(`${key}: ${Math.round(used[key] / 1024 / 1024 * 100) / 100} MB`);
  }
  
  // 内存使用超过阈值时触发垃圾回收
  if (used.heapUsed > 500 * 1024 * 1024) { // 500MB
    if (global.gc) {
      global.gc();
      console.log('触发垃圾回收');
    }
  }
};

// 定期监控内存
setInterval(monitorMemory, 30000);

连接池优化

js
// 数据库连接池
const { Pool } = require('pg');

const pool = new Pool({
  host: 'localhost',
  port: 5432,
  database: 'myapp',
  user: 'username',
  password: 'password',
  max: 20, // 最大连接数
  idleTimeoutMillis: 30000,
  connectionTimeoutMillis: 2000,
});

// 使用连接池
const getUser = async (id) => {
  const client = await pool.connect();
  try {
    const result = await client.query('SELECT * FROM users WHERE id = $1', [id]);
    return result.rows[0];
  } finally {
    client.release();
  }
};

性能监控与分析

关键性能指标

js
// 性能指标收集
class PerformanceMonitor {
  constructor() {
    this.metrics = {
      renderTime: [],
      memoryUsage: [],
      responseTime: []
    };
  }
  
  // 记录渲染时间
  recordRenderTime(startTime) {
    const renderTime = Date.now() - startTime;
    this.metrics.renderTime.push(renderTime);
    
    // 保持最近1000条记录
    if (this.metrics.renderTime.length > 1000) {
      this.metrics.renderTime.shift();
    }
  }
  
  // 获取平均渲染时间
  getAverageRenderTime() {
    const times = this.metrics.renderTime;
    return times.reduce((sum, time) => sum + time, 0) / times.length;
  }
  
  // 获取95百分位渲染时间
  get95thPercentileRenderTime() {
    const sorted = [...this.metrics.renderTime].sort((a, b) => a - b);
    const index = Math.floor(sorted.length * 0.95);
    return sorted[index];
  }
}

const monitor = new PerformanceMonitor();

// 在渲染中间件中使用
app.use((req, res, next) => {
  const startTime = Date.now();
  
  res.on('finish', () => {
    monitor.recordRenderTime(startTime);
  });
  
  next();
});

实时性能监控

js
// 性能监控仪表板
const express = require('express');
const app = express();

app.get('/metrics', (req, res) => {
  const metrics = {
    timestamp: new Date().toISOString(),
    memory: process.memoryUsage(),
    uptime: process.uptime(),
    averageRenderTime: monitor.getAverageRenderTime(),
    p95RenderTime: monitor.get95thPercentileRenderTime(),
    activeConnections: server.connections || 0
  };
  
  res.json(metrics);
});

// 健康检查端点
app.get('/health', (req, res) => {
  const health = {
    status: 'healthy',
    timestamp: new Date().toISOString(),
    uptime: process.uptime(),
    memory: process.memoryUsage().heapUsed / 1024 / 1024 // MB
  };
  
  // 检查内存使用是否过高
  if (health.memory > 1000) { // 1GB
    health.status = 'warning';
  }
  
  res.json(health);
});

实战案例:电商网站优化

项目背景

一个大型电商网站,包含商品列表、详情页、用户中心等功能,日 PV 100万+。

优化前的问题

  • 首屏加载时间 3.5s
  • 服务器 CPU 使用率 80%+
  • 内存使用 2GB+
  • 用户跳出率 45%

优化方案

1. 分层缓存策略

js
// 三级缓存架构
class CacheManager {
  constructor() {
    this.l1Cache = new Map(); // 内存缓存
    this.l2Cache = redis.createClient(); // Redis 缓存
    this.l3Cache = 'CDN'; // CDN 缓存
  }
  
  async get(key) {
    // L1: 内存缓存
    if (this.l1Cache.has(key)) {
      return this.l1Cache.get(key);
    }
    
    // L2: Redis 缓存
    const l2Data = await this.l2Cache.get(key);
    if (l2Data) {
      this.l1Cache.set(key, JSON.parse(l2Data));
      return JSON.parse(l2Data);
    }
    
    return null;
  }
  
  async set(key, value, ttl = 300) {
    // 设置到所有缓存层
    this.l1Cache.set(key, value);
    await this.l2Cache.setex(key, ttl, JSON.stringify(value));
  }
}

2. 智能预加载

js
// 基于用户行为的预加载
class SmartPreloader {
  constructor() {
    this.userBehavior = new Map();
  }
  
  // 记录用户行为
  recordBehavior(userId, action, data) {
    if (!this.userBehavior.has(userId)) {
      this.userBehavior.set(userId, []);
    }
    
    this.userBehavior.get(userId).push({
      action,
      data,
      timestamp: Date.now()
    });
  }
  
  // 预测用户下一步行为
  predictNextAction(userId) {
    const behaviors = this.userBehavior.get(userId) || [];
    const recent = behaviors.slice(-10); // 最近10次行为
    
    // 简单的预测逻辑
    const patterns = recent.reduce((acc, behavior) => {
      acc[behavior.action] = (acc[behavior.action] || 0) + 1;
      return acc;
    }, {});
    
    return Object.keys(patterns).reduce((a, b) => 
      patterns[a] > patterns[b] ? a : b
    );
  }
  
  // 预加载相关数据
  async preloadForUser(userId) {
    const nextAction = this.predictNextAction(userId);
    
    switch (nextAction) {
      case 'viewProduct':
        await this.preloadPopularProducts();
        break;
      case 'viewCategory':
        await this.preloadCategories();
        break;
    }
  }
}

3. 渐进式渲染

js
// 分块渲染策略
const renderProductPage = async (productId, req, res) => {
  // 立即发送页面框架
  res.write(`
    <!DOCTYPE html>
    <html>
    <head>
      <title>商品详情</title>
      <style>${criticalCSS}</style>
    </head>
    <body>
      <div id="header">...</div>
      <div id="product-container">
        <div class="loading">加载中...</div>
      </div>
  `);
  
  // 异步加载商品数据
  const product = await getProduct(productId);
  
  // 发送商品基本信息
  res.write(`
    <script>
      document.getElementById('product-container').innerHTML = \`
        <h1>${product.name}</h1>
        <div class="price">¥${product.price}</div>
        <div class="loading">加载更多信息...</div>
      \`;
    </script>
  `);
  
  // 异步加载详细信息
  const [reviews, recommendations] = await Promise.all([
    getProductReviews(productId),
    getRecommendations(productId)
  ]);
  
  // 发送完整内容
  res.write(`
    <script>
      // 更新页面内容
      updateProductDetails(${JSON.stringify({ reviews, recommendations })});
    </script>
    </body>
    </html>
  `);
  
  res.end();
};

优化结果

  • 首屏加载时间:3.5s → 1.2s(提升 66%)
  • 服务器 CPU 使用率:80% → 35%(降低 56%)
  • 内存使用:2GB → 800MB(降低 60%)
  • 用户跳出率:45% → 28%(降低 38%)

最佳实践总结

1. 缓存策略

  • 多层缓存:内存 + Redis + CDN
  • 智能失效:基于内容变化的缓存失效
  • 预热机制:系统启动时预加载热点数据

2. 代码优化

  • 按需加载:路由级和组件级代码分割
  • Tree Shaking:移除未使用的代码
  • Bundle 分析:定期分析包大小

3. 资源优化

  • 图片优化:WebP 格式、懒加载、响应式图片
  • 字体优化:字体子集、预加载、fallback
  • CSS 优化:关键 CSS 内联、非关键 CSS 异步加载

4. 监控与调试

  • 性能监控:实时监控关键指标
  • 错误追踪:完善的错误日志和报警
  • A/B 测试:验证优化效果

工具推荐

性能分析工具

  • Lighthouse:综合性能评估
  • WebPageTest:详细的加载分析
  • Chrome DevTools:实时性能调试

监控工具

  • New Relic:应用性能监控
  • DataDog:基础设施监控
  • Sentry:错误追踪

构建工具

  • Webpack Bundle Analyzer:包大小分析
  • Next.js Bundle Analyzer:Next.js 专用分析
  • Rollup Plugin Visualizer:Rollup 包分析

总结

构建高性能的 SSR 应用需要在多个层面进行优化:

  1. 缓存策略:合理的缓存可以显著提升性能
  2. 代码分割:减少初始加载时间
  3. 资源优化:优化图片、字体、CSS 等资源
  4. 服务器优化:提升服务器处理能力
  5. 监控分析:持续监控和优化

记住,性能优化是一个持续的过程,需要根据实际业务场景和用户反馈不断调整策略。最重要的是要建立完善的监控体系,以数据驱动优化决策。

参考资源

vitepress开发指南