2025上半年,最新 AI实践都在这!20+ 应用案例,任听一场议题就值回票价 了解详情
写点什么

听说 Signals 快要登陆 React 了?

  • 2024-04-22
    北京
  • 本文字数:2727 字

    阅读完需:约 9 分钟

听说Signals快要登陆React了?

“Signals”专门用于管理客户端状态,而且从最近的趋势来看,其很有可能在 React 中发挥作用。



就在上周,Dashi Kato(Waku 的缔造者)发布了 use-signals,一个面向 TC39 signals 的实验性 React hook,旨在演示 Signals 如何在 React 中发挥作用。



Signals 是什么?


Signals 已然经历约 10 年的发展周期,先后得到 Angular、Vue、SolidJS、Qwik 以及 Preact 等众多 JavaScript 框架的采纳,用于管理客户端状态。Signals 属于能自动跟踪其使用位置的变量。一旦 Signal 发生变更,其值就会失效,进而触发 UI 状态更新 / 重新渲染。


举例来说,这是一个名为 counter 的 Signal,其值为 0。


const counter = new Signal.State(0);
复制代码


需要明确的是,Signals 与 React 的 useState 有着本质区别。useState 是 React 提供的 hook,用于管理功能组件内的状态,并允许开发者声明状态变量并更新该变量的函数。


Signals 则是事件的侦听器或者观察器,用于处理异步事件或是超出组件直接控制之外的数据变更。因此,大家会看到 Signal 声明中并没有定义“setter”函数。但配合 React 之后,情况将大为不同。下面来看之前的 Signal 如何在 React 中进行声明:


const [counter, setCounter] = useState(0);
复制代码


Signals 的概念之所以非常有趣,就是因为 React 那“自上而下”的模型意味着每当有状态值发生变化时,组件树的所有后代都会重新渲染并对 UI 执行相应变更,从而保证 DOM/UI 与应用程序的状态同步。


而在使用 Signals 管理状态之后,开发者能够以更细粒度方式控制对 UI 中的哪些部分进行“重新渲染”。但不要误会,这并不是说 Signals 比 React 方法的性能更高,只是二者的功能定位有所不同。


Signals 使用方法


如上所示,我们可以使用 new 构造函数来声明一个 Signal,例如:


const counter = new Signal.State(0);
复制代码


之后,要“get”一个 Signal 的值,我们可以使用.get() 方法;而要“set”或者更新一个 Signal,我们可以使用.set() 方法。例如:


counter.get();counter.set(10);
复制代码


Signals 在 React 中如何起效?


跟之前提到的 Signals 使用方法不同,它在 React 中另有起效方式。绕过 React 的 diffing 无疑有违 React 声明式编程这一核心原则,因此 React 中的 Signals 仍将使用 VDOM,而且同样会像变更 useState 那样触发重新渲染。


那么在 React 中使用 Signals 还有何意义?其实我的第一反应也是如此,但请大家先别急,让我们一同探索 Signals 的深邃世界。


TC39 提案


如果 TC39 提案获得通过,则意味着 Signals 将在 JavaScript 中原生可用,届时我们将可以在框架之外使用 Signals。更重要的是,框架作者们理论上也能够以标准化方式实现 Signals。换言之,Signals 机制的任何改进都将令所有采用标准化方法的框架受益。而目前,大部分使用 Signals 的框架所采取的处理方式都各有细微差异。


4 月 11 日,Rob Eisenberg 宣布 TC39 提案已经进入第一阶段。这意味着该提案已被纳入 ECMAScript 的考量范围。虽然还有很多工作要做,但这项提案似乎正朝着正确的方向稳步迈进。



TC39 提案还强调了围绕不同框架特定要求的方式开发 API 的重要性。use-signals 的意义也正在于此,它在使用建议 Signals API 的同时,也仍然遵循 React 的核心设计原则。


无论 React 团队最终是否会采纳 Signals,use-signals 的出现都在一定程度上表明 Signals 确实能够在 React 中发挥作用。


Signal Utils 提案


Signals 目前仅支持原语,但也有其他 signal-utils 提案正在推进当中,努力将对象和数组引入其中。例如:


对象


import { SignalObject } from 'signal-utils/object';

let obj = new SignalObject({ isLoading: true, error: null, result: null,});
复制代码


数组


import { SignalArray } from 'signal-utils/array';

let arr = new SignalArray([1, 2, 3]);
复制代码


在 React 中使用 Signals 示例


聊了这么多,下面我们一起来看在 React 中具体如何使用 Signals。这里展示的 React 代码在 Waku 上下文中运行,默认在服务器端进行渲染,但其也能支持纯客户端组件的“use client”指令。


与 useState 不同的是,对 Signals 来说,新的 Signals 声明是在组件外部实现的。在以下代码片段当中,也就是名为 counter 的 const,其用于存储初始值并可以被传递至 useSignal hook。


该 count const 公开了.set() 与.get() 方法,这些方法可以在事件处理函数 handleInc 当中使用。


Signal 值的变更将触发 DOM 更新,并在 UI 中显示新的 count 值。


尤其有趣的一点是,在返回的 Jsx 当中,我们不再需要使用.get() 来访问并显示 HTML  <div/> 元素中的值。相反,现在可以直接访问 count 值。这与 Qwik 的实现略有不同,后者需要我们从 counter 处访问值,例如 counter.value。


'use client';

import { Signal, useSignal } from 'use-signals';

const counter = new Signal.State(0);

const UseSignalComponent = () => { const count = useSignal(counter);

const handleInc = () => counter.set(counter.get() + 1);

return ( <> <div>useSignal count: {count}</div> <button type='button' onClick={handleInc}> Update count </button> </> );};

export default UseSignalComponent;
复制代码


useState 组件


为了直观比较,以下代码的业务逻辑不变、只是用 React 的 useState hook 进行编写。可以看到,二者之间几乎没有区别。


'use client';

import { useState } from 'react';

const UseStateComponent = () => { const [count, setCount] = useState(0);

const handleInc = () => setCount(count + 1);

return ( <> <div>useState count: {count}</div> <button type='button' onClick={handleInc}> Update count </button> </> );};

export default UseStateComponent;
复制代码


混合 Signals


总结来讲,在 React 中实现 Signals 完全具备可行性。虽然 Signals 可能需要一段时间才能在 JavaScript 中获得原生身份,但我个人高度赞赏其蓬勃发展的技术社区对于全新开发方式的探索。感兴趣的朋友不妨持续关注 GitHub repo:++dai-shi/use-signals++。


原文链接


https://thenewstack.io/did-signals-just-land-in-react/


声明:本文为 InfoQ 翻译整理,未经许可禁止转载。


今日好文推荐


德国再次拥抱Linux:数万系统从windows迁出,能否避开二十年前的“坑”?


谷歌大裁员引发元老集体抗议:领导脑袋空空,无能的中层管理团队不断扩大


系统 bug 致百人入狱,砸了 2.8 亿元仍上云失败!二十年了,这家大企业被日本软件坑惨了


“真男人就应该用 C 编程”!用 1000 行 C 代码手搓了一个大模型,Mac 即可运行,特斯拉前 AI 总监爆火科普 LLM


2024-04-22 17:243226

评论

发布
暂无评论
发现更多内容

LeetCode题解:127. 单词接龙,BFS+生成所有可能新单词再匹配,JavaScript,详细注释

Lee Chen

算法 大前端 LeetCode

升级redhat6的yum源替换为centos源

Bruce Xiong

让你的产品更懂世界:如何提升场景文本识别中的语言模型

华为云开发者联盟

神经网络 文字 语义

“懂行”为舟,助互联网驶向“新蓝海”

一次资源泄露问题排查纪录

AI乔治

Java 架构 JVM 内存泄漏

在算力“沃土”上,种植互联网下一个奇迹十年

9年技术面试官讲解:计算机专业应届生怎样写简历

Java架构师迁哥

游戏服务器多钱一个月呢?

德胜网络-阳

程序员什么时候就该辞职了?

Java架构师迁哥

架构师训练营第 1 期第 13 周作业

好吃不贵

极客大学架构师训练营

看完这份文档我吊打了BATJ面试官,他问的我全都会:Spring+逻辑算法+MySQL+Java+Redis+并发编程+JVM+RabbitMQ等

Java架构之路

Java 程序员 架构 面试 编程语言

太狠了阿里技术专家撰写的电子版JVM&G1 GC实战,颠覆了传统认知

Java架构之路

Java 程序员 架构 面试 编程语言

Techo | 大数据专场报名盛启!12月20日欢迎莅临!

腾讯云大数据

大数据 数据仓库

ROS 机器人操作系统进阶实战

Geek_3cc3ec

Python 人工智能 学习 ROS

Java 并发编程:volatile能否保证数据的同步

李尚智

Java Java并发

盘点2020 | 技术圈里的这些热名词

xcbeyond

微服务 DDD 金融科技 新基建 盘点2020

2020年第11期公有云性能评测:盛大云-华东实现“三冠”,百度云虎视眈眈

博睿数据

百度云 腾讯云 阿里云 公有云 华为云

synchronized 是王的后宫总管,线程是王妃

Java架构师迁哥

校园恋爱新技能:智慧琴房恋爱助攻手册

IoT云工坊

物联网 API sdk 智慧琴房 智慧校园

【变与不变】架构中的边界划定

soolaugust

编程 架构 设计

你确定你会算数吗?老大说:你连这个都不知道还敢面试电商公司?

小Q

Java 学习 编程 程序员 面试

朋友不讲武德急催我给他Java干货教程,我劝他耗子尾汁并丢给他一份GitHub上标星115k+的Java教程,他看了之后连忙向我道歉!

Java架构之路

Java 程序员 架构 面试 编程语言

官方活动 | 日更挑战(初阶)——七日更,挑战百元京东E卡!

InfoQ写作社区官方

七日更 热门活动

免费分享!GitHub标星15k的Java编程思想最新中文版,肝了一周整理成1539页的PDF文档!

Java架构之路

Java 程序员 架构 面试 编程语言

A Guide for Accidental Project Managers

Geek_ed0696

pmp project manager role of PMP PMP Certification greycampus

芯片破壁者(二十二):政府与半导体间的“美国往事”

脑极体

架构师训练营第 1 期第 13 周学习总结

好吃不贵

极客大学架构师训练营

ES6中的Promise和Generator详解

程序那些事

新特性 ES6 Promise 程序那些事 Generator

“懂行人”携手互联网,创造未来无限可能

字节跳动总监亲自整理,在知乎高达5716赞的Java开发手记。

Java架构之路

Java 程序员 架构 面试 编程语言

云智一体:攀登2021智能经济新山峰

脑极体

听说Signals快要登陆React了?_架构/框架_Paul Scanlon_InfoQ精选文章