前端框架进化史

article/2025/8/5 3:32:07

本内容是对 You’ll Never Manually Update the DOM Again // Here’s Why 内容的翻译与整理。

你再也不需要手工更新DOM, 以下是原因

现代 JavaScript 框架,如 React、Vue、Svelte、Solid、Quick,以及本周推出的其他 786 个框架,都试图做一些不同的事情。

React 为我们提供了虚拟 DOM 中的组件。Vue 使响应式编程感觉自然。Svelte 抛弃了运行时,所有内容都在编译时处理。Solid 跟踪细粒度更新,而 Quick 则尝试暂停和恢复整个应用程序。但在它们所有炫酷的语法糖和闪亮特性背后,它们都围绕着一个奇妙的技巧展开,那就是响应式 UI 更新

那么,什么是响应式 UI 更新呢?这意味着你不再需要一步步地告诉浏览器如何更改 UI,而是声明基于某些状态 UI 应该是什么样子。框架会以最有效的方式来实现这一点,这说起来很不错,因为在 2000 年代初期,JavaScript 框架尚未存在的 VanillaJS 时代,你必须手动完成所有事情:

如果想更新 UI,你得找到元素,改变它的文本内容,可能添加一个类,可能删除一个,甚至可能重绘整个区块,只为显示一项新内容。如果状态在两个地方发生变化,你必须更新两个地方,你会遇到奇怪的闪烁或过时的 UI,或者那个拒绝更新的 div,直到你刷新页面,而它仍然只在你哭了一会儿之后才会更新。

我不知道这是否是个必要条件,但这确实是过程的一部分。因此,要构建任何体面的东西,你必须查询 DOM 中的元素。你必须在某处手动跟踪应用状态。你必须编写命令式代码,以反映每个状态变化在 UI 中的体现。你必须在播放林肯公园和穿着工装短裤的同时,调试 Internet Explorer 6。你必须完成所有这些,尤其是最后一点,否则它将不同步,显示过时的数据,而你甚至不会知道,直到用户给你发邮件,或者产品经理在午夜时分通过 Slack 给你发了截图和带有消极攻击性的“嗯?”。不过,当时 Slack 甚至还不存在。所以,我们迫切需要更好的方法。这里的“我们”并不是指我,我当时才 10 岁。

然后,jQuery 出现了。我们所需要的英雄,终于来了。突然间,你可以这样写一行代码,这意味着不再需要使用 document.getElementById,也不再有跨浏览器的头痛,不再在 Netscape 中哭泣。jQuery 给了我们力量,给了我们简单。短暂的时间内,我们认为它是未来,它确实感觉像未来。但问题是,它仍然是命令式的。你仍然需要控制何时更新,在哪里更新,所有的负担依旧,只是以更简单的形式呈现。你的状态可能会变化,如果你忘记在某处调用 jQuery 更新函数,那么你的 UI 又会不同步。老实说,曾经干净的 JavaScript 代码库现在变成了一片意大利面森林,充满了 .Hide、.show, .addClass、.removeClass 和偶尔的存在危机。

因此,jQuery 让你能够更快更轻松地操控 DOM,但 Angular 1.0 想:如果你根本不需要考虑 DOM 呢?他们的解决方案,无论你是否喜欢,就是“算了,让我们随时监视一切。”Angular 的脏检查意味着框架会在每次发生事件时检查页面上的每一个变量,看看是否有变化。听着,这对我们制作的数千个待办应用来说是个好主意,但在构建真正动态的网站时就没那么好了,因为应用变得很慢,可以想象,非常慢。它确实检查了所有内容。开发人员们开始意识到,我们不仅需要一种注意变化的方法,我们还需要对变化做出反应的方式,就像你需要对 2025 年 AI 编写越来越多代码的变化做出反应一样。

(这段为软广) 因为现在我们不仅要对 UI 问题做出反应,还要对安全问题做出反应。这就是我与 Sneak 合作的原因,Sneak 是一个开发者安全平台,帮助开发人员发现、修复和跟踪代码中的漏洞。为什么你应该关注 2025 年 5 月 28 日举行的 Sneak Launch。那是一个免费的虚拟活动,讨论 AI 如何改变应用开发,以及这对安全意味着什么。你将听到如何处理新兴的 AI 威胁。因此,不仅是你的代码由 AI 编写,或者其他人用 AI 编写的代码,还有人利用 AI 来扩展 AI 生成代码的安全性。调整你的工作流程,以适应这一切,构建一个稳固的应用安全治理。无论你是使用 React 构建,还是发布 LLM 功能,或者只是让 AI 自动补全一半的代码库,风险都是真实的。风险一直存在,但现在比以往任何时候都更为严重。这次活动有两场会议,一场在上午 10 点,另一场在下午 6 点。所以没有借口,无论你的工作时间表或时区如何。只需点击描述中的链接注册 Sneak Launch 2025。再次强调,活动将于 5 月 28 日举行,免费并且面向开发人员。重大变化即将来临,确保你为此做好准备。

因此,开发人员不仅想注意变化,他们还想对变化做出反应,于是你猜对了,React 出现了。React 的回答是重新渲染一切,但首先在内存中完成。每次发生变化时,React 都不是直接更新实际的 DOM,而是构建虚拟 DOM 的副本。虚拟 DOM 通过比较新旧副本来找出变化,并只更新必要的部分。这比脏检查有了巨大的改进,但仍然存在开销,因为你需要处理整个树。这就是其他框架的出现,想:如果我们也不这样做呢?例如 Svelte,Svelte 在编译时处理这一切。换句话说,他们在构建步骤中分析你的代码并将所有内容连接起来。它已经知道这个 H1 标签依赖于 name。所以当 name 改变时,Svelte 只更新那个 DOM 节点。没有虚拟 DOM,没有差异计算,仅仅是精确的编译 DOM 更新。但这有一个权衡,那就是你失去了一些灵活性,但换来了速度和简单性。

Vue 则采取了与这两者不同的方法。它不是使用信号或编译,而是将你的状态封装在代理中。因此,当你访问一个属性时,Vue 会跟踪它。当你更新它时,Vue 知道该重渲染什么。就像 Vue 在暗中监视你的变量,但以一种有帮助的方式。你得到了所需的响应性,但没有虚拟 DOM 差异计算的开销。仅基于实际使用的内容进行高效的 DOM 补丁。

而 Solid 也完全跳过虚拟 DOM,但采用了基于信号的方法。Solid 中的每个响应式状态都是一个信号。当信号变化时,只有依赖于它的 DOM 节点更新,别无其他,而我实际上非常喜欢这一点。这非常快,老实说相当优雅。你改变一个信号,就像是打开一个开关,只有灯光响应。

然后是 Quick,它想:如果我们在用户做出反应之前从未运行我们的应用程序呢?因为 Quick 在服务器上序列化你的应用程序,发送零 JavaScript,并且只有在用户与之交互时才恢复执行。它仍然是响应式的,但现在优化了水合性能和冷启动,这对于边缘渲染或需要瞬时感的大型应用来说是理想的。

那么,所有这些归根结底,为什么要使用响应式 UI 更新呢?因为所有这些框架,无论它们的方法有多不同,都在尝试解决同一个问题:保持 UI 与状态同步,并高效、可靠且具有装饰性地完成这一切。响应式 UI 更新是实现这一目标的基础。它消除了手动 DOM 同步,让你专注于 UI 应该是什么样子,而不必手动完成每一件事情。它可以从计数器扩展到完整的应用程序。它们都有不同的哲学和权衡,有些比其他的更快。

但每个现代 JavaScript 框架都是响应式的,因为手动处理实在太糟糕,我们再也不想回头。我希望你喜欢这个视频,并对一些 JavaScript 框架有了更深入的了解。我认为,理解我们在技术方面的来源,以及分解那些抽象层,确实有助于我们整体上成为更好的开发者。因此,如果你同意这一点,可以看看这两个视频中的一个,也许它们也能给你带来帮助。或者通过订阅频道等我的下一个视频。


http://www.hkcw.cn/article/EkMMXiCeiC.shtml

相关文章

Redis笔记

Redis(Remote Dictionary Server),开源、基于C语言、内存可持久化的NoSQL的键值对数据库。 命令:redis命令不区分大小写,set和SET效果相同 主键(key):任意二进制序列(字…

flask pyinstaller打包exe,出现module not found问题

最近大作业要做一个项目要打包成可执行程序,这里说一下这个module not found问题,并提供几种可能的方案,如果严格按照这些来走就能解决常见问题,剩下的神仙问题建议问问ai或者清缓存重试 首先说一下目录问题,这应该是包括我(打包app.py)在内的大多数人遇见该报错问题的原因,提…

基于SpringBoot+Redis实现RabbitMQ幂等性设计,解决MQ重复消费问题

一、实现方案 本实验方案参考「RabbitMQ消息可靠性深度解析|从零构建高可靠消息系统的实战指南」 1、业务层幂等处理: 每个消息携带一个全局唯一ID,在业务处理过程中,首先检查这个ID是否已经被处理过。例如,将已处理消…

性能优化 - 案例篇:数据一致性

文章目录 Pre引言1. 分布式缓存概念2. Redis 与 Memcached 区别概览3. Spring Boot 中使用 Redis3.1 引入依赖与常用客户端3.2 RedisTemplate 的基本用法3.3 Spring Cache 注解式缓存 4. 秒杀业务简介及挑战5. Lua 脚本实现原子库存扣减5.1 准备阶段:数据预加载5.2 …

【深度学习】 19. 生成模型:Diffusion Models

Diffusion Models Diffusion Models 简介 Diffusion 模型是一类通过逐步添加噪声并再逆向还原的方式进行图像生成的深度生成模型。其基本流程包括: 前向过程(Forward Process):将真实图像逐步加噪,最终变为高斯噪声…

【速通RAG实战:进阶】22、RAG 技术前沿探索:GraphRAG 等 13 种技术详解与应用场景

一、RAG技术的演进脉络与前沿分类 (一)从基础RAG到前沿创新的技术跃迁 传统RAG(检索增强生成)通过“检索-生成”两阶段解决LLM的知识时效性和准确性问题,但在复杂推理、多模态融合、成本控制等场景面临瓶颈。前沿RAG技术围绕检索精度、推理深度、生成质量、系统效率四大…

美业新动能:智能体如何赋能行业“维护”升级(3/6)

摘要:美业行业蓬勃发展,但竞争激烈、客户要求提高等挑战并存。智能体技术应运而生,它融合机器学习、自然语言处理和计算机视觉等技术,实现精准营销、个性化服务,优化客户关系、设备和供应链维护。本文探讨智能体在美业…

RAGflow详解及实战指南

目录 前言 一、RAGflow核心技术解析 1. 技术原理:检索与生成的协同进化 2. 架构设计:分层模块化与高扩展性 3. 核心优势:精准、高效、安全 二、RAGflow实战应用场景 1. 企业知识库搭建 2. 智能客服系统 3. 投资分析报告生成 4. 制造…

C# winform教程(二)

一、基础控件 常用的基础控件主要有按钮,文本,文本输入,组,进度条,等等。 基础控件 名称含义详细用法Button按钮checkbox多选按钮Combobox下拉选择groupbox组控件label标签,显示文字panel控件集合&#xf…

Altium Disigner(16.1)学习-原理图绘制以及必要操作

一、下载软件 通过网盘分享的文件:Altium Designer 16.zip 链接: https://pan.baidu.com/s/1uBHeoJJ-iA2tXw3NRjCcdA?pwd7c3h 提取码: 7c3h 复制这段内容后打开百度网盘手机App,操作更方便哦 --来自百度网盘超级会员v5的分享 二、建立工程 添加proje…

DAY 18 推断聚类后簇的类型

目录 DAY 18 推断聚类后簇的类型1.推断簇含义的2个思路:先选特征和后选特征2.通过可视化图形借助ai定义簇的含义3.科研逻辑闭环:通过精度判断特征工程价值作业:参考示例代码对心脏病数据集采取类似操作,并且评估特征工程后模型效果有无提升。…

常见的PLC浮点数字节序转换方法

变量的字节序 在 PLC 中,寄存器的长度通常为 16 bit,常见的数据类型有 16bit、32bit长度的 对于 32 bit 长度的数据,比如浮点型(在西门子、Codesys 中称为 REAL 型),由于长度较长,在不同平台、…

【Day42】

DAY 42 Grad-CAM与Hook函数 知识点回顾 回调函数lambda函数hook函数的模块钩子和张量钩子Grad-CAM的示例 作业:理解下今天的代码即可 """ Day 42: Grad-CAM与Hook函数本节主要内容: 1. 回调函数(Callback)和lambda函数- 回调函数是作为参…

各种乱码问题解决措施

1.乱码问题原因 1.数据的编码和解码使用的不是同一个字符集 编码就是我们能看懂的数据转换为计算机能够识别的二进制编码 解码就是将存在计算机里面的二进制编码的文件转化为我们能够识别的字符等内容。 2.使用了不支持某个语言文字的字符集。 1,1html文件乱码 在<meta c…

c++ 异常处理

测试代码&#xff1a; #include <exception>void testException() { // test exceptionclass MyException : public exception {public:using std::exception::exception; // 关键&#xff1a;继承父类所有构造函数};struct Divide {int operator() (int a, int b) cons…

yolo个人深入理解

卷积层的理解,通过云端服务器训练模型,模型构建的重要性,针对极低像素的处理,模型训练召回率提高技巧,卷积层2,4,8,16,32的小模型与大模型的理解 一.关于backbone,neck,head深入理解 1,backbone的主要组成部分是sppf和conv,这是backbone的核心,其中yolov5和yolov8…

Git基本操作

目录 1. 创建Git本地仓库 2. 配置Git 3. 认识工作区、暂存区、版本库 4. 添加文件 -- 场景一 5. 查看.git文件 6. 添加文件 -- 场景二 7. 修改文件 8. 版本回退 9. 撤销修改 9.1 情况一&#xff1a;对于工作区的代码&#xff0c;还没有add 9.2 情况二&#xff1a;已…

《中国棒垒球》注册青少年运动员需要什么条件·棒球1号位

青少年注册棒球/垒球运动员&#xff1a; 基础条件 | Basic Requirements 年龄范围 通常6-18岁&#xff08;按U8/U10/U12/U15/U18分组&#xff09; Typically 6-18 years old (grouped as U8/U10/U12/U15/U18) 健康证明 需提交体检报告&#xff0c;重点关注心肺功能与运动损…

华为深度学习面试手撕题:手写nn.Conv2d()函数

题目 只允许利用numpy包&#xff0c;实现Pytorch二维卷积函数nn.Conv2d() 解答 此代码考察二维卷积的概念&#xff0c;详见&#xff1a; 6.2. 图像卷积 — 动手学深度学习 2.0.0 documentation 6.3. 填充和步幅 — 动手学深度学习 2.0.0 documentation 6.4. 多输入多输出通…

MMR 最大边际相关性详解

最大边际相关性&#xff08;MMR&#xff0c;max_marginal_relevance_search&#xff09;的基本思想是同时考量查询与文档的 相关度&#xff0c;以及文档之间的 相似度。相关度 确保返回结果对查询高度相关&#xff0c;相似度 则鼓励不同语义的文档被包含进结果集。具体来说&…