Angular ng-state script 元素的生成机制介绍
ng-state
的生成过程是在 Angular SSR 中非常关键的部分。为了让客户端能够接管服务器渲染的页面状态,Angular 在服务器端需要将应用的当前状态保存下来,并将其嵌入到返回的 HTML 中。这样,客户端在接管时就可以直接使用这些状态,而不必重新发起 API 请求或重新计算数据。
这个状态的传递是通过 ng-state
script 标签实现的,里面包含了整个应用的序列化状态信息,通常是以 JSON 格式存储。通过这个脚本标签,Angular 在客户端执行时可以“水合”(hydrate)这些状态信息,继续执行剩下的逻辑。
生成过程的核心步骤
在应用执行 SSR 时,Angular 会经历多个阶段,最终生成包含 ng-state
的 HTML 响应。我们可以通过以下几个方面来理解 ng-state
的生成过程:
-
服务器端初始化
当用户请求一个 Angular SSR 页面时,服务器端的 Angular 应用会先初始化。这包括启动 Angular 的服务器端应用模块 (AppServerModule
) 并解析当前请求的路由。服务器端会加载与该路由相关的组件,同时请求相关的数据,比如 API 调用或数据库查询。举个例子,一个电子商务网站的用户请求了首页,服务器端会初始化对应的模块,并请求首页所需的产品数据。在这个过程中,Angular SSR 会像在客户端一样初始化组件,并使用 Angular 的依赖注入系统来加载数据服务和状态管理工具。
-
数据获取与处理
当服务器端应用加载完成时,任何需要通过外部 API 获取的数据都会被请求。比如,一个博客页面可能会请求最新的文章列表,一个电子商务页面可能会请求产品详情。这些数据会通过 Angular 的HttpClient
服务获取。服务器端在完成这些请求后,会将数据存储在 Angular 应用的状态管理工具(如 NgRx)或本地组件的变量中。
-
状态的序列化
当所有的数据获取和处理都完成后,Angular 会进入渲染阶段。在渲染过程中,服务器端应用会将所有的状态数据序列化成 JSON 格式。这些状态包括页面所需的所有动态数据,比如用户信息、API 响应、表单数据等。序列化的过程非常类似于 JavaScript 中的
JSON.stringify
,Angular 会通过这种方式将应用的状态对象转化为 JSON 格式,以便嵌入到返回的 HTML 中。对于复杂的应用来说,这个序列化过程可能涉及大量的数据结构和对象。 -
插入
ng-state
标签
服务器端渲染完成后,Angular 会将生成的 HTML 发送给客户端。在这个 HTML 中,ng-state
脚本标签被插入到页面的底部,通常位于关闭</body>
标签之前。这个脚本标签的内容是之前序列化的 JSON 数据,它是整个应用当前状态的快照。这个标签的形式如下:
<script id="ng-state" type="application/json"> {"books": [{ "id": 1, "title": "Angular in Action", "author": "John Doe" },{ "id": 2, "title": "Pro Angular", "author": "Jane Doe" },{ "id": 3, "title": "Learning Angular", "author": "Jim Beam" }],"totalBooks": 3 } </script>
在这个例子中,
ng-state
保存了书籍列表和总数,当客户端 Angular 启动时,它会从这个 JSON 中恢复状态。 -
客户端水合
当 HTML 响应到达客户端后,浏览器会首先渲染 HTML 内容。此时,页面已经显示出来,用户可以看到初始的内容,而 Angular 应用在客户端还没有真正启动。Angular 在客户端启动时,会检测页面中是否存在
ng-state
标签。检测到后,Angular 会从这个标签中读取 JSON 对象,并将其还原为应用的状态。接着,客户端应用就能够继续使用这些状态信息,避免重新发起数据请求。例如,用户看到的首页中的书籍列表,已经通过 SSR 渲染出来,而客户端 Angular 启动时,它会从
ng-state
标签中获取相同的书籍数据,并将这些数据加载到客户端的状态管理工具中。
具体的案例分析
在一个典型的博客网站中,服务器端渲染的过程中可能需要获取文章列表并将其传递给客户端。假设我们在服务器端使用 Angular SSR 来渲染这个博客页面,并且通过 Angular 的 HttpClient
发起 API 请求获取文章列表。
-
用户请求博客页面
当用户请求https://example.com/blog
时,Angular 的服务器端应用会处理这个请求,并加载博客页面的组件。服务器端会通过HttpClient
发起 API 请求,获取当前的文章列表。 -
服务器端渲染文章列表
获取到文章列表后,服务器端会将这些数据传递给 Angular 组件,并进行渲染。组件会将文章列表渲染为 HTML,同时,文章列表的数据会被序列化并存储到ng-state
标签中。 -
生成的 HTML
服务器端返回给客户端的 HTML 可能是这样的:<html> <head><title>博客首页</title> </head> <body><div id="content"><h1>最新文章</h1><ul><li>文章1: 如何使用 Angular SSR</li><li>文章2: 深入理解 NgRx</li><li>文章3: Angular Router 高级技巧</li></ul></div><script id="ng-state" type="application/json">{"articles": [{ "id": 1, "title": "如何使用 Angular SSR", "author": "张三" },{ "id": 2, "title": "深入理解 NgRx", "author": "李四" },{ "id": 3, "title": "Angular Router 高级技巧", "author": "王五" }]}</script> </body> </html>
在这个例子中,HTML 内容已经包含了文章列表,用户可以立即看到这些内容,而
ng-state
标签则保存了相同的文章列表数据,以供客户端使用。 -
客户端恢复状态
当客户端 Angular 应用启动时,它会检测到页面中存在ng-state
标签,并从中获取文章列表数据。然后,客户端 Angular 应用会将这些数据恢复到本地状态管理工具(比如 NgRx store 或者服务中的变量)。这样,用户的体验是无缝的。页面内容已经由服务器端渲染并显示,客户端应用加载后,继续使用服务器端提供的数据,而不需要再次发起 API 请求。这显著提升了加载性能,尤其是对于内容密集型的应用。
现实中的应用场景
实际应用中,ng-state
的生成与使用不仅仅限于博客或电子商务网站,几乎任何需要服务器端渲染的应用都可以受益于这个机制。比如:
-
内容管理系统(CMS)
对于一个 CMS 网站,编辑和用户访问的内容是高度动态的,而 SSR 可以加速页面加载。通过ng-state
,编辑在服务器端创建或修改的内容可以立即被传递到客户端,避免了客户端的重复加载和数据获取。 -
在线零售
电子商务平台通常会展示大量产品列表和详情。在服务器端渲染产品页面时,通过ng-state
可以将产品信息、库存状态、价格等数据一次性传递给客户端,减少不必要的 API 请求,并且确保页面在用户访问时立即显示。 -
社交媒体
社交媒体平台的动态更新是另一个使用ng-state
的好例子。用户的时间线数据、朋友列表、通知等信息都可以在服务器端渲染,并通过ng-state
传递给客户端,确保用户在页面加载时就可以看到最新的内容,而不是等待数据的重新获取。
小结
ng-state
的生成过程是 Angular SSR 机制中关键的一步,通过它,服务器端渲染生成的状态数据可以被序列化并传递给客户端。这个过程不仅加速了页面的加载速度,还减少了服务器端与客户端之间的冗余请求,为用户提供了更好的体验。在实际应用中,开发者需要根据应用的需求,合理使用 ng-state
来传递状态,保证应用的流畅性和一致性。
通过这个机制,Angular 能够在现代 web 应用中实现更好的性能优化,同时保持复杂的交互和状态管理。这使得 Angular SSR 成为许多高性能网站的首选技术。
相关文章:

Angular ng-state script 元素的生成机制介绍
ng-state 的生成过程是在 Angular SSR 中非常关键的部分。为了让客户端能够接管服务器渲染的页面状态,Angular 在服务器端需要将应用的当前状态保存下来,并将其嵌入到返回的 HTML 中。这样,客户端在接管时就可以直接使用这些状态,…...

小程序-全局数据共享
目录 1.什么是全局数据共享 2. 小程序中的全局数据共享方案 MboX 1. 安装 MobX 相关的包 2. 创建 MobX 的 Store 实例 3. 将 Store 中的成员绑定到页面中 4. 在页面上使用 Store 中的成员 5. 将 Store 中的成员绑定到组件中 6. 在组件中使用 Store 中的成员 1.什么是全…...

vSAN01:vSAN简介、安装、磁盘组、内部架构与调用关系
目录 传统的共享存储vSAN存储OSA的系统要求vSAN安装vSAN集群vSAN skyline healthvSAN与HA磁盘组混合磁盘架构全闪磁盘架构 vSAN对象vSAN内部架构 传统的共享存储 通过隔离的存储网络使得不同的ESXi主机访问独立的存储设备。需要前期投入较高的资金单独采购存储、网络可以单独规…...

Apache NiFi最全面试题及参考答案
目录 解释什么是Apache NiFi以及它的主要用途。 NiFi 的数据处理流程是怎样的? NiFi 的架构包括哪些组件? 解释 NiFi 的 “FlowFile” 概念及其组成部分。 NiFi 的 “Processor” 是什么?有哪些类型? 如何在 NiFi 中创建一个新的数据流? NiFi 的 “Connection” 有…...

基于Docker部署最新版本SkyWalking【10.1.0版本】
文章目录 前言前置条件一、创建Docker 网络二、部署 SkyWalking OAP 服务器三 部署 SkyWalking UI四 查看日志4.1. 查看 SkyWalking OAP 日志4.2. 查看 SkyWalking UI 日志 五 停止并删除容器结论 前言 由于本地的 JDK 版本与 SkyWalking 对应的 JDK 版本不一致,为…...

如何在 Ubuntu 18.04 上使用 LEMP 安装 WordPress
前些天发现了一个巨牛的人工智能学习网站,通俗易懂,风趣幽默,忍不住分享一下给大家。点击跳转到网站。 简介 WordPress 是互联网上最流行的 CMS(内容管理系统)。它允许您在 MySQL 后端和 PHP 处理的基础上轻松设置灵…...

shadcn-vue 快速入门(2)
components.json 关于项目配置 components.json 文件保存了项目的配置信息。 我们使用该文件了解项目的基本设定,并生成定制化的组件以适应项目需求。 注意:components.json 文件是可选的,仅在使用 CLI 向项目添加组件时才需要。如果使用复…...

Oracle数据恢复—异常断电导致Oracle数据库报错的数据恢复案例
Oracle数据库故障: 机房异常断电后,Oracle数据库启库报错:“system01.dbf需要更多的恢复来保持一致性,数据库无法打开”。数据库没有备份,归档日志不连续。用户方提供了Oracle数据库的在线文件,需要恢复zxf…...

数据结构-4.1.特殊矩阵的压缩存储
一.一维数组的存储结构: 1.知道一维数组的起始地址,就可以求出任意下标对应的元素所在的地址; 2.注:如果数组下标从1开始,上述公式的i就要改为i-1; 3.数组里的元素类型相同,因此所占空间也相同…...

Hive数仓操作(十四)
一、Hive的DDL语句 在 Hive 中,DDL(数据定义语言)语句用于数据库和表的创建、修改、删除等操作。以下是一些重要的 DDL 语句: 1. 创建数据库和表 创建数据库 CREATE DATABASE IF NOT EXISTS database_name;创建表 CREATE TABLE …...

SpringBoot技术:实现古典舞在线交流平台的秘诀
摘 要 随着互联网技术的发展,各类网站应运而生,网站具有新颖、展现全面的特点。因此,为了满足用户古典舞在线交流的需求,特开发了本古典舞在线交流平台。 本古典舞在线交流平台应用Java技术,MYSQL数据库存储数据&#…...

自动驾驶系列—全面解析自动驾驶线控制动技术:智能驾驶的关键执行器
🌟🌟 欢迎来到我的技术小筑,一个专为技术探索者打造的交流空间。在这里,我们不仅分享代码的智慧,还探讨技术的深度与广度。无论您是资深开发者还是技术新手,这里都有一片属于您的天空。让我们在知识的海洋中…...

YOLO11改进|卷积篇|引入可变核卷积AKConv
目录 一、AKConv卷积1.1AKConv卷积介绍1.2AKConv核心代码 五、添加MLCA注意力机制5.1STEP15.2STEP25.3STEP35.4STEP4 六、yaml文件与运行6.1yaml文件6.2运行成功截图 一、AKConv卷积 1.1AKConv卷积介绍 AKConv允许卷积参数的数量以线性方式增加或减少,而不是传统的…...

推荐 uniapp 相对好用的海报生成插件
插件地址:自定义canvas样式海报 - DCloud 插件市场 兼容性也是不错的:...

MySQL表操作(进阶)
一、数据库约束 1、约束类型 NOT NULL - 指示某列不能存储 NULL 值 UNIQUE - 保证某列的每行必须有唯一的值 DEFAULT - 规定没有给列赋值时的默认值 PRIMARY KEY - NOT NULL 和 UNIQUE 的结合。确保某列(或两个列多个列的结合)有唯一标 识ÿ…...

【设计模式】软件设计原则——接口隔离迪米特
接口隔离原则引出 接口隔离原则 定义:用多个专门的接口,不使用单一的总接口,客户端不应该依赖它不需要的接口; 一个类对另一个类的依赖,应该建立在最小接口上;如果有一个大接口,里面有很多方法,如果使用一个类实现该接口,所有的类都要实现,导致代码冗余;…...

【C++】——list的介绍和模拟实现
P. S.:以下代码均在VS2019环境下测试,不代表所有编译器均可通过。 P. S.:测试代码均未展示头文件stdio.h的声明,使用时请自行添加。 博主主页:Yan. yan. …...

B树系列解析
我最近开了几个专栏,诚信互三! > |||《算法专栏》::刷题教程来自网站《代码随想录》。||| > |||《C专栏》::记录我学习C的经历,看完你一定会有收获。||| > |||《Linux专栏》࿱…...

docker 部署 WEB IDE
简介 问题描述:GitCode 的 Web IDE 不满足个人使用需求 如何解决:在本机或云服务器部署 Web IDE 如何解决 拉取容器镜像 docker pull coder/code-server 运行 docker run -d --name vscode -p 8080:8080 -p 8443:8443 -e PASSWORD"123456&quo…...

【Android】数据存储
本章介绍Android五种主要存储方式的用法,包括共享参数SharedPreferences、数据库SQLite、SD卡文件、App的全局内存,另外介绍重要组件之一的应用Application的基本概念与常见用法,以及四大组件之一的内容提供器ContentProvider的基本概念与常见…...

个人网络安全的几个重点与防御
1 浏览器 firefox 这是第一选择 如果你真的不明白可以找找各个浏览器漏洞 mail 的危险的 来自与代理和漏洞 浏览器溢出漏洞 实时注意更新就可以 2 防火墙 大家都用windows 只需在 gpedit.msc 设置 但有什么未知漏洞就不得而知了 因为美国的计划问题 网络端口溢出漏洞 但…...

python爬虫 - 初识爬虫
🌈个人主页:https://blog.csdn.net/2401_86688088?typeblog 🔥 系列专栏:https://blog.csdn.net/2401_86688088/category_12797772.html 目录 前言 一、爬虫的关键概念 (一)HTTP请求与响应 ࿰…...

tomcat版本升级导致的umask问题
文章目录 1、问题背景2、问题分析3、深入研究4、umask4.1、umask的工作原理4.2、umask的计算方式4.3、示例4.4、如何设置umask4.5、注意事项 1、问题背景 我们的java服务是打成war包放在tomcat容器里运行的,有一天我像往常一样去查看服务的日志文件,却提…...

Golang | Leetcode Golang题解之第455题分发饼干
题目: 题解: func findContentChildren(g []int, s []int) (ans int) {sort.Ints(g)sort.Ints(s)m, n : len(g), len(s)for i, j : 0, 0; i < m && j < n; i {for j < n && g[i] > s[j] {j}if j < n {ansj}}return }...

vscode+stfp插件,实现远程自动同步文件代码
概述 远程同步代码,将本地代码实时保存到同一局域网内的另一台电脑(linux系统),这里的本地代码也可以是远程服务上的代码,即从一个远程ip同步到另一台远程ip服务器。 工具 vscode,SFTP插件 安装 vscod…...

python 实现djb2哈希算法
djb2哈希算法介绍 DJB2哈希算法是一种简单且快速的哈希算法,由Daniel J. Bernstein设计。这种算法的实现非常简单,适用于短键值的哈希表,也常被用于嵌入式设备和资源受限的系统。 基本原理 DJB2算法的原理是将输入的字符串视为一个字节数组…...

文件夹作为普通文件而非子模块管理
relaxed_ik_ros2 文件夹下存在 .gitmodules 文件和 .gitignore 文件。这说明该目录已经被 Git 识别为子模块。 要将这个文件夹作为普通文件而非子模块管理,你可以按以下步骤操作: 1. 删除子模块配置 首先删除 .gitmodules 文件中的子模块配置。你可以…...

7c结构体
文章目录 一、结构体的设计二、结构体变量的初始化2.1结构体在内存表示;**2.2**结构体类型声明和 结构体变量的定义和初始化只声明结构体类型声明类型的同时定义变量p1用已有结构体类型定义结构体变量p2*定义变量的同时赋初值。*匿名声明结构体类型 2.3 结构体嵌套及…...

浅聊前后端分离开发和前后端不分离开发模式
1.先聊聊Web开发的开发框架Spring MVC 首先要知道,Spring MVC是Web开发领域的一个知名框架,可以开发基于请求-响应模式的Web应用。而Web开发的本质是遵循HTTP(Hyper Text Transfer Protocol: 超文本传输协议)协议【发请求…...

RabbitMQ篇(死信交换机)
目录 一、简介 二、TTL过期时间 三、应用场景 一、简介 当一个队列中的消息满足下列情况之一时,可以成为死信(dead letter) 消费者使用basic.reject或者basic.nack声明消费失败,并且消息的requeue参数设置为false消息是一个过…...