前端框架进化史
本内容是对 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 框架有了更深入的了解。我认为,理解我们在技术方面的来源,以及分解那些抽象层,确实有助于我们整体上成为更好的开发者。因此,如果你同意这一点,可以看看这两个视频中的一个,也许它们也能给你带来帮助。或者通过订阅频道等我的下一个视频。
相关文章:

前端框架进化史
本内容是对 You’ll Never Manually Update the DOM Again // Here’s Why 内容的翻译与整理。 你再也不需要手工更新DOM, 以下是原因 现代 JavaScript 框架,如 React、Vue、Svelte、Solid、Quick,以及本周推出的其他 786 个框架,都试图做一些…...
“轻量应用服务器” vs. “云服务器CVM”:小白入门腾讯云,哪款“云机”更适合你?(场景、配置、价格对比解析)
更多云服务器知识,尽在hostol.com 当你第一次踏入腾讯云这个“数字百货大楼”,面对琳琅满目的“云产品”,是不是有点眼花缭乱,特别是看到“轻量应用服务器”和“云服务器CVM”这两位都号称能帮你“安家落户”的“云主机”时&…...
day63—回溯—全排列(LeetCode-46)
题目描述 给定一个不含重复数字的数组 nums ,返回其 所有可能的全排列 。你可以 按任意顺序 返回答案。 示例 1: 输入:nums [1,2,3] 输出:[[1,2,3],[1,3,2],[2,1,3],[2,3,1],[3,1,2],[3,2,1]]示例 2: 输入&#x…...

(二)stm32使用4g模块(移远ec800k)连接mqtt
下面代码是随手写的,没有严谨测试仅供参考测试 uint8_t msgBuf[200]{"msg from mcu"}; uint8_t txBuf[250]{0}; uint16_t msgid0; uint16_t mqttTaskState0; uint16_t t100msCount0; uint8_t sendFlag10; uint8_t sendFlag20; void t100msTask1(void) { …...

防火墙iptables项目实战
目录 一、网络规划 三、环境准备与检测 1、firewall (1)配置防火墙各大网卡ip并禁用firewalld和selinux (2)打开firewall路由转发 2、PC1(内网) (1)配置ip并禁用firewalld和s…...

webpack继续学习
认识PostCSS工具 PostCSS是一个通过JS来转换样式的工具,这个工具可以帮助我们进行一些CSS的转换和适配,比如自动添加浏览器前缀,css样式的重置 实现这些功能需要借助于PostCSS对应的插件 自动添加浏览器前缀需要: npm install…...

Scrapy爬虫框架Spiders爬虫脚本使用技巧
我们都知道Scrapy是一个用于爬取网站数据、提取结构化数据的Python框架。在Scrapy中,Spiders是用户自定义的类,用于定义如何爬取某个(或某些)网站,包括如何执行爬取(即跟踪链接)以及如何从页面中…...

PowerBI企业运营分析—全动态盈亏平衡分析
PowerBI企业运营分析—全动态盈亏平衡分析 欢迎来到Powerbi小课堂,在竞争激烈的市场环境中,企业运营分析平台成为提升竞争力的核心工具。 该平台通过整合多源数据,实现关键指标的实时监控,从而迅速洞察业务动态,精准…...

docker的基本命令
容器的三大组成 镜像image 一个静态文件,特点:分层结构,不可更改 容器container 镜像运行的结果,容器可以修改,运行完后直接停止 仓库registry 用来存放镜像文件的地方 容器的常用命令介绍 关于镜像的命令 docker …...

【运维实战】Rsync将一台主Web服务器上的文件和目录同步到另一台备份服务器!
在管理 Web 服务器时,确保数据安全且在发生故障时能够快速恢复至关重要,备份和镜像 Web 服务器数据最可靠的方法之一是使用 rsync。 Rsync 工具可以帮助在两台服务器之间同步文件和目录,非常适合用于创建 Web 服务器数据的备份和镜像。 下面…...
实时通信RTC与传统直播的异同
实时通信(RTC)与直播虽然在音视频传输领域密切相关,但设计目标和实现原理是存在显著差异的。 一、核心联系 共同目标:均需实现音视频数据的采集、编码、传输与播放。技术重叠:使用相似的编码标准(如H.264/…...
Python-正则表达式(re 模块)
目录 一、re 模块的使用过程二、正则表达式的字符匹配1. 匹配开头结尾2. 匹配单个字符3. 匹配多个字符4. 匹配分组5. Python 代码示例 三、re 模块的函数1. 函数一览表2. Python 代码示例1)search 与 finditer2)findall3)sub4)spl…...

AgenticSeek 本地部署教程(Windows 系统)
#工作记录 Fosowl/agenticSeek:完全本地的 Manus AI。 部署排错参考资料在文末 或查找往期笔记。 AgenticSeek 本地部署教程(Windows 系统) 一、环境准备 1. 安装必备工具 Docker Desktop 下载地址:Docker Desktop 官网 安装后启…...

基于 qiankun + vite + vue3 构建微前端应用实践
核心内容摘要 技术栈组合 采用 Vite Vue3 Qiankun 构建微前端架构主应用和子应用独立开发部署,通过 Qiankun 集成 2. 主应用关键配置通过 registerMicroApps 注册子应用,配置路由匹配规则(activeRule)使用…...
VR教育:开启教育新时代的钥匙
VR 教育,即虚拟现实教育,是将虚拟现实技术(Virtual Reality,简称 VR)应用于教育领域的一种创新教育模式。它借助计算机技术、图形图像技术、传感器技术等,创建出高度逼真的虚拟学习环境,让学生通过头戴式显示设备、手柄…...

机器学习:逻辑回归与混淆矩阵
本文目录: 一、逻辑回归Logistic Regression二、混淆矩阵(一)精确率precision(二)召回率recall(三)F1-score:了解评估方向的综合预测能力(四)Roc曲线…...

20250602在荣品的PRO-RK3566开发板的Android13下打开HDMI显示
20250602在荣品的PRO-RK3566开发板的Android13下打开HDMI显示 2025/6/2 16:20 缘起:貌似荣品的PRO-RK3566开发板的Android13默认关闭了HDMI显示。 据说:荣品确认RK3566的GPU比较弱,同时开【MIPI接口的】LCD屏显示和HDMI显示容易出现异常。 更…...
【学习记录】快速上手 PyQt6:设置 Qt Designer、PyUIC 和 PyRCC 在 PyCharm中的应用
文章目录 📌 前言✅ 第一步:安装 PyQt6 及其工具包🔧 第二步:找到相关工具路径🧰 第三步:在 PyCharm 中配置外部工具打开设置🛠️ 配置 Qt Designer🛠️ 配置 PyUIC6(UI转…...

AI在网络安全领域的应用现状和实践
当前,人工智能技术已深度融入网络安全产品,推动传统防御模式向智能化、自适应方向加速演进。各安全厂商通过机器学习、深度学习与知识图谱等技术的融合应用,提高安全产品在威胁检测、攻击溯源、风险评估等场景的能力跃迁,突破传统…...
DrissionPage 异常处理实战指南:构建稳健的网页自动化防线
在网页自动化领域,异常处理能力直接决定了系统的健壮性。作为融合Selenium与Requests特性的创新工具,DrissionPage提供了多层次的异常处理机制。本文将深入剖析其异常体系,结合真实场景案例,为您构建一套完善的自动化容错方案。 …...

鸿蒙任务项设置案例实战
目录 案例效果 资源文件与初始化 string.json color.json CommonConstant 添加任务 首页组件 任务列表初始化 任务列表视图 任务编辑页 添加跳转 任务目标设置模型(formatParams) 编辑页面 详情页 任务编辑列表项 目标设置展示 引入目标…...

TDengine 的 AI 应用实战——运维异常检测
作者: derekchen Demo数据集准备 我们使用公开的 NAB数据集 里亚马逊 AWS 东海岸数据中心一次 API 网关故障中,某个服务器上的 CPU 使用率数据。数据的频率为 5min,单位为占用率。由于 API 网关的故障,会导致服务器上的相关应用…...
DHCP与DNS的配置
在网络管理中,DHCP(动态主机配置协议)和DNS(域名系统)是两个关键组件。DHCP用于自动分配IP地址,而DNS用于将域名解析为IP地址。本文将详细介绍如何在Linux环境下配置DHCP和DNS服务。 一、DHCP配置 1. 安装…...

使用Plop.js高效生成模板文件
前情 开发是个创造型的职业,也是枯燥的职业,因为开发绝大多数都是每天在业务的代码中无法自拨,说到开发工作,就永远都逃不开新建文件的步骤,特别现在组件化开发胜行,每天都是在新建新建组件的道路上一去不…...

Vue框架2(vue搭建方式2:利用脚手架,ElementUI)
一.引入vue第二种搭建方式 在以前的前端项目中,一个项目需要多个html文件实现页面之前的切换,如果页面中需要依赖js或者css文件,那么我们就需要在多个html文件中都需要导入vue.js文件,太过繁琐. 现在前端开发都采用单页面结果,一个项目中只有一个html文件 其他不同的内容都写…...

mac 设置cursor (像PyCharm一样展示效果)
一、注册 Cursor - The AI Code Editor 二、配置Python环境 我之前使用pycharm创建的python项目,以及创建了虚拟环境,现在要使用cursor继续开发。 2.1 选择Python 虚拟环境 PyCharm 通常将虚拟环境存储在项目目录下的 venv 或 .venv 文件夹中…...

SpringCloudAlibaba微服务架构
技术架构图 SpringCloudAlibaba微服务架构 说明: 1.1、采用SpringCloudAlibaba分布式微服务架构,使用Nginx做代理,服务治理使用Nacos组件,Gateway网关做权限验证、路由、过滤。 1.2、Redis做消息缓存,包括数据大屏、数…...

Java高级 | 【实验三】Springboot 静态资源访问
隶属文章: Java高级 | (二十二)Java常用类库-CSDN博客 系列文章: Java高级 | 【实验一】Spring Boot安装及测试 最新-CSDN博客 Java高级 | 【实验二】Springboot 控制器类相关注解知识-CSDN博客 目录 一、Thymeleaf 1.1 是什么&…...
C语言_预处理详解
1. 预定义符号 C语言设置了一些预定义符号,可以直接使用,预定义符号也是在预处理期间处理的 1 __FILE__ //进行编译的源文件 2 __LINE__//文件当前的行号 3 __DATE__ //文件被编译的日期 4 __TIME__//文件被编译的时间 5 __STDC__//如果编译器遵循ANSI…...
将前后端分离版的前端vue打包成EXE的完整解决方案
将若依前后端分离版的前端打包成EXE的完整解决方案 将若依前后端分离版的Vue前端打包成Windows可执行文件(.exe),同时保持与后端API的通信,需要使用Electron框架来实现。下面是详细的步骤和解决方案。 一、准备工作 1. 环境要求 Node.js (推荐v16+)npm 或 yarn若依前后端分…...