一文探究Angular中的服务端渲染(SSR)

一文探究Angular中的服务端渲染(SSR)

一般来说,普通的 Angular 应用是在 浏览器 中运行,在 DOM 中对页面进行渲染,并与用户进行交互。而 Angular Universal 是在 服务端 进行渲染(Server-Side Rendering,SSR),生成静态的应用程序网页,然后在客户端展示,好处是可以更快地进行渲染,在提供完整的交互之前就可以为用户提供内容展示。【相关教程推荐:《angular教程》】

本文是在 Angular 14 环境中完成,有些内容对于新的 Angular 版本可能并不适用,请参考 Angular 官方文档。

使用 SSR 的好处

对 SEO 更加友好

虽然现在包括 Google 在内的某些搜索引擎和社交媒体声称已经能支持对由 JavaScript(JS)驱动的 SPA(Single-Page Application)应用进行爬取,但是结果似乎差强人意。静态 HTML 网站的 SEO 表现还是要好于动态网站,这也是 Angular 官网所持有的观点(Angular 可是 Google 的!)。

Universal 可以生成无 JS 的静态版本的应用程序,对搜索、外链、导航的支持更好。

提高移动端的性能

某些移动端设备可能不支持 JS 或者对 JS 的支持非常有限,导致网站的访问体验非常差。这种情况下,我们需要提供无 JS 版本的应用,以便为用户提供更好的体验。

更快地展示首页

对于用户的使用体验来说,首页展示速度的快慢至关重要。根据 eBay 的数据,搜索结果的展示速度每提高 100 毫秒,“添加至购物车”的使用率就提高 0.5%。

使用了 Universal 之后,应用程序的首页会以完整的形态展示给用户,这是纯的 HTML 网页,即使不支持 JS,也可以展示。此时,网页虽然不能处理浏览器的事件,但是支持通过 routerLink 进行跳转。

这么做的好处是,我们可以先用静态网页抓住用户的注意力,在用户浏览网页的时候,同时加载整个 Angular 应用。这给了用户一个非常好的极速加载的体验。

为项目增加 SSR

Angular CLI 可以帮助我们非常便捷的将一个普通的 Angular 项目转变为一个带有 SSR 的项目。创建服务端应用只需要一个命令:

ng add @nguniversal/express-engine

登录后复制

建议在运行该命令之前先提交所有的改动。

这个命令会对项目做如下修改:

添加服务端文件:

main.server.ts – 服务端主程序文件app/app.server.module.ts – 服务端应用程序主模块tsconfig.server.json – TypeScript 服务端配置文件server.ts – Express web server 的运行文件

修改的文件:

package.json – 添加 SSR 所需要的依赖和运行脚本angular.json – 添加开发、构建 SSR 应用所需要的配置

在 package.json 中,会自动添加一些 npm 脚本:dev:ssr 用于在开发环境运行 SSR 版本;serve:ssr 用于直接运行 build 或 prerender 后的网页;build:ssr 构建 SSR 版本的网页;prerender 构建预渲染后的网页,与 build 不同,这里会根据提供的 routes 生成这些页面的 HTML 文件。

替换浏览器 API

由于 Universal 应用不是在浏览器中执行,因此一些浏览器的 API 或功能将不可用。例如,服务端应用是无法使用浏览器中的全局对象 window、document,navigator,location。

Angular 提供了两个可注入对象,用于在服务端替换对等的对象:Location 和 DOCUMENT。

例如,在浏览器中,我们通过 window.location.href 获取当前浏览器的地址,而改成 SSR 之后,代码如下:

import { Location } from '@angular/common'; export class AbmNavbarComponent implements OnInit{  // ctor 中注入 Location  constructor(private _location:Location){    //...  }   ngOnInit() {    // 打印当前地址    console.log(this._location.path(true));  }}

登录后复制

同样,对于在浏览器使用 document.getElementById() 获取 DOM 元素,在改成 SSR 之后,代码如下:

import { DOCUMENT } from '@angular/common'; export class AbmFoxComponent implements OnInit{  // ctor 中注入 DOCUMENT  constructor(@Inject(DOCUMENT) private _document: Document) { }   ngOnInit() {    // 获取 id 为 fox-container 的 DOM    const container = this._document.getElementById('fox-container');  }}

登录后复制

使用 URL 绝对地址

在 Angular SSR 应用中,HTTP 请求的 URL 地址必须为 绝对地址(即,以 http/https 开头的地址,不能是相对地址,如 /api/heros)。Angular 官方推荐将请求的 URL 全路径设置到 renderModule() 或 renderModuleFactory() 的 options 参数中。但是在 v14 自动生成的代码中,并没有显式调用这两个方法的代码。而通过读 Http 请求的拦截,也可以达到同样的效果。

下面我们先准备一个拦截器,假设文件位于项目的 shared/universal-relative.interceptor.ts 路径:

import { HttpHandler, HttpInterceptor, HttpRequest } from '@angular/common/http';import { Inject, Injectable, Optional } from '@angular/core';import { REQUEST } from '@nguniversal/express-engine/tokens';import { Request } from 'express'; // 忽略大小写检查const startsWithAny = (arr: string[] = []) => (value = '') => {    return arr.some(test => value.toLowerCase().startsWith(test.toLowerCase()));}; // http, https, 相对协议地址const isAbsoluteURL = startsWithAny(['http', '//']); @Injectable()export class UniversalRelativeInterceptor implements HttpInterceptor {    constructor(@Optional() @Inject(REQUEST) protected request: Request) { }     intercept(req: HttpRequest, next: HttpHandler) {        // 不是绝对地址的 URL        if (!isAbsoluteURL(req.url)) {            let protocolHost: string;            if (this.request) {                // 如果注入的 REQUEST 不为空,则从注入的 SSR REQUEST 中获取协议和地址                protocolHost = `${this.request.protocol}://${this.request.get(                    'host'                )}`;            } else {                // 如果注入的 REQUEST 为空,比如在进行 prerender build:                // 这里需要添加自定义的地址前缀,比如我们的请求都是从 abmcode.com 来。                protocolHost = 'https://www.abmcode.com';            }            const pathSeparator = !req.url.startsWith('/') ? '/' : '';            const url = protocolHost + pathSeparator + req.url;            const serverRequest = req.clone({ url });            return next.handle(serverRequest);         } else {            return next.handle(req);        }    }}

登录后复制

然后在 app.server.module.ts 文件中 provide 出来:

import { UniversalRelativeInterceptor } from './shared/universal-relative.interceptor';// ... 其他 imports@NgModule({  imports: [    AppModule,    ServerModule,    // 如果你用了 @angular/flext-layout,这里也需要引入服务端模块    FlexLayoutServerModule,   ],  providers: [    {      provide: HTTP_INTERCEPTORS,      useClass: UniversalRelativeInterceptor,      multi: true    }  ],  bootstrap: [AppComponent],})export class AppServerModule { }

登录后复制

这样任何对于相对地址的请求都会自动转换为绝对地址请求,在 SSR 的场景下不会再出问题。

Prerender 预渲染静态 HTML

经过上面的步骤后,如果我们通过 npm run build:ssr 构建项目,你会发现在 dist//browser 下面只有 index.html 文件,打开文件查看,发现其中还有 这样的元素,也就是说你的网页内容并没有在 html 中生成。这是因为 Angular 使用了动态路由,比如 /product/:id 这种路由,而页面的渲染结果要经过 JS 的执行才能知道,因此,Angular 使用了 Express 作为 Web 服务器,能在服务端运行时根据用户请求(爬虫请求)使用模板引擎生成静态 HTML 界面。

而 prerender(npm run prerender)会在构建时生成静态 HTML 文件。比如我们做企业官网,只有几个页面,那么我们可以使用预渲染技术生成这几个页面的静态 HTML 文件,避免在运行时动态生成,从而进一步提升网页的访问速度和用户体验。

预渲染路径配置

需要进行预渲染(预编译 HTML)的网页路径,可以有几种方式进行提供:

通过命令行的附加参数:

ng run :prerender --routes /product/1 /product/2

登录后复制

如果路径比较多,比如针对 product/:id 这种动态路径,则可以使用一个路径文件:

routes.txt

/products/1/products/23/products/145/products/555

登录后复制

然后在命令行参数指定该文件:

ng run :prerender --routes-file routes.txt

登录后复制

在项目的 angular.json 文件配置需要的路径:

 "prerender": {   "builder": "@nguniversal/builders:prerender",   "options": {     "routes": [ // 这里配置       "/",       "/main/home",       "/main/service",       "/main/team",       "/main/contact"     ]   },

登录后复制

配置完成后,重新执行预渲染命令(npm run prerender 或者使用命令行参数则按照上面中的命令执行),编译完成后,再打开 dist//browser 下的 index.html 会发现里面没有 了,取而代之的是主页的实际内容。同时也生成了相应的路径目录以及各个目录下的 index.html 子页面文件。

SEO 优化

SEO 的关键在于对网页 title,keywords 和 description 的收录,因此对于我们想要让搜索引擎收录的网页,可以修改代码提供这些内容。

在 Angular 14 中,如果路由界面通过 Routes 配置,可以将网页的静态 title 直接写在路由的配置中:

{ path: 'home', component: AbmHomeComponent, title: '' },

登录后复制

另外,Angular 也提供了可注入的 Title 和 Meta 用于修改网页的标题和 meta 信息:

import { Meta, Title } from '@angular/platform-browser'; export class AbmHomeComponent implements OnInit {   constructor(    private _title: Title,    private _meta: Meta,  ) { }   ngOnInit() {    this._title.setTitle('');    this._meta.addTags([      { name: 'keywords', content: '' },      { name: 'description', content: '' }    ]);  }}

登录后复制

总结

Angular 作为 SPA 企业级开发框架,在模块化、团队合作开发方面有自己独到的优势。在进化到 v14 这个版本中提供了不依赖 NgModule 的独立 Component 功能,进一步简化了模块化的架构。

Angular Universal 主要关注将 Angular App 如何进行服务端渲染和生成静态 HTML,对于用户交互复杂的 SPA 并不推荐使用 SSR。针对页面数量较少、又有 SEO 需求的网站或系统,则可以考虑使用 Universal 和 SSR 技术。

更多编程相关知识,请访问:编程教学!!

以上就是一文探究Angular中的服务端渲染(SSR)的详细内容,更多请关注【创想鸟】其它相关文章!

版权声明:本文内容由互联网用户自发贡献,该文观点仅代表作者本人。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如发现本站有涉嫌抄袭侵权/违法违规的内容, 请发送邮件至253000106@qq.com举报,一经查实,本站将立刻删除。

发布者:PHP中文网,转转请注明出处:https://www.chuangxiangniao.com/p/2696972.html

(0)
上一篇 2025年3月7日 18:37:09
下一篇 2025年2月23日 23:01:20

AD推荐 黄金广告位招租... 更多推荐

相关推荐

  • 一文聊聊Angular中的管道(PIPE)

    什么是管道(pipe)?本篇文章给大家介绍一下angular中的管道(pipe),聊聊内置管道和自定义管道的方法,希望对大家有所帮助! 什么是管道(PIPE) PIPE,翻译为管道。Angular 管道是编写可以在HTML组件中声明的显示值…

    2025年3月7日 编程技术
    200
  • 一文详解Angular父子组件间传数据的方法

    本篇文章带大家了解一下angular父子组件(component)之间传递数据的方法,介绍一下angular中父组件向子组件传数据、子组件向父组件传数据的方法,希望对大家有所帮助! 环境:Angular CLI: 11.0.6Angular…

    2025年3月7日
    200
  • angular怎么进行样式隔离?实现机制详解

    angular怎么进行样式隔离?下面本篇文章就来带大家了解一下angular的样式隔离实现机制,希望对大家有所帮助! angular 以组件为基本单位。我们编写一个一个的组件,再将这些组件组合为一颗组件树。但是在开发的过程中,经常需要在父组…

    2025年3月7日
    200
  • 浅析Angular中的自定义结构型/属性型指令

    angular指令分为三种,组件(带模板指令)、结构型指令(改变宿主文档结构)、属性型指令(改变宿主行为),下面主要介绍自定义结构型指令和自定义属性型指令。 一、自定义结构型指令 一个元素上只能放一个结构型指令,结构型指令的书写形式为*指令…

    2025年3月7日
    200
  • Angular学习之聊聊生命周期

    本篇文章带大家继续angular的学习,使用angular进行开发时,避免不了需要接触生命周期,下面就来带大家一起聊聊angular中的生命周期,希望对大家有所帮助! 接触过 react 和 vue 开发的读者应该对生命周期这个概念不陌生。…

    2025年3月7日 编程技术
    200
  • angular学习之深入聊聊状态和动画

    本篇文章带大家深入了解一下angular中的状态和动画,简单介绍一下创建动画的方法,并聊聊关键帧动画、动画回调、可重用动画、交错动画等知识点,希望对大家有所帮助! 状态 1、什么是状态 状态表示的是要进行运动的元素在运动的不同时期所呈现的样…

    2025年3月7日 编程技术
    200
  • Angular学习之ControlValueAccessor接口详解

    controlvalueaccessor 是什么?为什么需要使用 ?下面本篇文章就来带大家了解angular中的controlvalueaccessor组件接口,希望对大家有所帮助! ControlValueAccessor 是什么? 简单…

    2025年3月7日
    200
  • 构建你的第一个Angular应用:存储和访问数据

    在本系列的第一个教程中,我们学习了如何开始创建 Angular 应用程序。成功完成该教程后,您现在应该拥有第一个正常运行的 Angular 应用程序,其标题为“关于国家/地区的有趣事实”。在创建任何可以在屏幕上呈现的组件之前,我们将创建一些…

    2025年3月7日
    200
  • JS框架有哪些

    JS框架有哪些,需要具体代码示例 随着前端开发的发展,JavaScript(简称JS)框架成为了开发者不可或缺的工具。它们可以提供强大的功能,简化开发流程,并提高开发效率。本文将介绍几个常用的JS框架,并提供具体的代码示例供读者参考。 Re…

    2025年3月7日
    200
  • jQuery所需的依赖包有哪些?

    jQuery是一款非常受欢迎的JavaScript库,它能够简化DOM操作、事件处理、动画效果等操作。为了使用jQuery,我们需要在项目中引入相应的依赖包。下面将介绍一下jQuery引入所需的依赖包以及具体的代码示例。 首先,我们需要下载…

    2025年3月7日
    200

发表回复

登录后才能评论