Skip to content
⚠️ This article was written in 2021. Some content may be outdated.

Angular Universal 服務端渲染:SEO 與首屏優化完整指南

Angular Universal 讓 Angular 應用可以在服務端預渲染 HTML,解決兩個核心問題:SEO(搜索引擎抓取 SPA 困難)和首屏性能(FCP 指標優化)。Angular 12 的 Universal 已經相當成熟,這篇文章覆蓋從集成到優化的完整流程。

快速集成

bash
ng add @nguniversal/express-engine

# 生成的文件:
# ├── server.ts                  # Express 服務器
# ├── src/app/app.server.module.ts  # 服務端 AppModule
# └── src/main.server.ts         # 服務端入口

生成的 server.ts 開箱即用,無需大量配置:

typescript
// server.ts(簡化版)
import { ngExpressEngine } from "@nguniversal/express-engine";
import { AppServerModule } from "./src/main.server";

const app = express();

app.engine("html", ngExpressEngine({ bootstrap: AppServerModule }));
app.set("view engine", "html");
app.set("views", distFolder);

// 靜態資源
app.get("*.*", express.static(distFolder, { maxAge: "1y" }));

// SSR 路由
app.get("*", (req, res) => res.render("index", { req }));

服務端與瀏覽器環境的差異

SSR 最大的坑是:服務端沒有 windowdocumentlocalStorage 等瀏覽器 API。

typescript
import { isPlatformBrowser, isPlatformServer } from "@angular/common";
import { PLATFORM_ID } from "@angular/core";

@Injectable({ providedIn: "root" })
export class ThemeService {
  constructor(@Inject(PLATFORM_ID) private platformId: Object) {}

  getTheme(): string {
    if (isPlatformBrowser(this.platformId)) {
      return localStorage.getItem("theme") || "light";
    }
    return "light"; // 服務端默認值
  }

  setTheme(theme: string) {
    if (isPlatformBrowser(this.platformId)) {
      localStorage.setItem("theme", theme);
    }
  }
}

TransferState:避免雙重數據請求

SSR 的典型問題:服務端已經請求了 API 數據,但客户端 hydration 時又請求一次。TransferState 解決這個問題:

typescript
import { TransferState, makeStateKey } from "@angular/platform-browser";

const USERS_KEY = makeStateKey<User[]>("users");

@Injectable({ providedIn: "root" })
export class UserService {
  constructor(
    private http: HttpClient,
    private transferState: TransferState,
    @Inject(PLATFORM_ID) private platformId: Object,
  ) {}

  getUsers(): Observable<User[]> {
    // 客户端:先檢查 TransferState 緩存
    if (isPlatformBrowser(this.platformId)) {
      const cached = this.transferState.get(USERS_KEY, null);
      if (cached) {
        this.transferState.remove(USERS_KEY);
        return of(cached); // 直接使用服務端傳來的數據,不重複請求
      }
    }

    return this.http.get<User[]>("/api/users").pipe(
      tap((users) => {
        // 服務端:將數據注入 TransferState,傳遞給客户端
        if (isPlatformServer(this.platformId)) {
          this.transferState.set(USERS_KEY, users);
        }
      }),
    );
  }
}

Meta 標籤動態設置(SEO 關鍵)

typescript
@Component({
  template: `<h1>{{ post.title }}</h1>
    <p>{{ post.content }}</p>`,
})
export class PostDetailComponent implements OnInit {
  post: Post;

  constructor(
    private route: ActivatedRoute,
    private meta: Meta,
    private title: Title,
  ) {}

  ngOnInit() {
    this.route.data.subscribe(({ post }) => {
      this.post = post;

      // 設置頁面標題和 SEO meta 標籤
      this.title.setTitle(post.title);
      this.meta.updateTag({ name: "description", content: post.excerpt });
      this.meta.updateTag({ property: "og:title", content: post.title });
      this.meta.updateTag({
        property: "og:description",
        content: post.excerpt,
      });
      this.meta.updateTag({ property: "og:image", content: post.coverImage });
    });
  }
}

構建和部署

bash
# 構建 SSR 版本
npm run build:ssr

# 本地測試
npm run serve:ssr

# 部署到 Node.js 服務器
# dist/ 目錄包含:
# ├── browser/   # 客户端靜態資源(CDN 可用)
# └── server/    # Node.js 服務器代碼

性能優化建議

typescript
// server.ts - 添加頁面緩存
const cache = new Map<string, string>();

app.get("*", (req, res) => {
  const key = req.url;
  if (cache.has(key)) {
    return res.send(cache.get(key));
  }

  res.render("index", { req }, (err, html) => {
    if (!err) cache.set(key, html); // 緩存渲染結果
    res.send(html);
  });
});

SSR vs 預渲染

如果頁面內容是靜態的(博客、文檔),用預渲染(Prerendering)比 SSR 更簡單:

bash
# 構建時預渲染指定路由
npm run prerender

# angular.json 配置預渲染路由
{
  "prerender": {
    "routes": ["/", "/about", "/blog/1", "/blog/2"]
  }
}

預渲染產物是純靜態 HTML,可以直接部署到 CDN,無需 Node.js 服務器。

總結

Angular Universal 的集成成本在 Angular 12 已經相當低——ng add 一行命令就能搭好骨架。關鍵是要處理好平台差異(isPlatformBrowser)和 TransferState 數據傳遞。對內容型頁面(博客、商品詳情)來説,SSR 或預渲染帶來的 SEO 和首屏性能提升非常可觀。

MIT Licensed