React Native 全新架构来了
React Native 0.76 现已在 npm 上以全新架构默认发布!在 0.76 版本的发布博客中,我们分享了此版本包含的一系列庞大更改。在本文中,我们将概述全新架构以及它如何塑造 React Native 的未来。
全新架构全面支持今世 React 功能,包括 Suspense、Transitions、automatic batching 和 useLayoutEffect。全新架构还包括新的原生模块和原生组件体系,使您能够编写范例安全的代码,直接访问原生接口,无需桥接。
此次发布是自 2018 年以来我们从零开始重写 React Native 的成果,我们特殊注意使全新架构对于大多数应用来说是一个渐进式迁移过程。2021 年,我们建立了全新架构工作组,与社区互助,确保整个 React 生态体系的升级体验顺遂进行。
大多数应用将能够以与其他任何版本相同的积极级别采用 React Native 0.76。最盛行的 React Native 库已经支持全新架构。全新架构还包括一个主动互操作层,以实现与针对旧架构的库的向后兼容性。
在过去几年的开发过程中,我们的团队公开分享了对全新架构的愿景。假如您错过了其中的任何讲座,请在以下链接查看:
[*]React Native EU 2019 - 新的 React Native
[*]React Conf 2021 - React 18 主旨演讲
[*]App.js 2022 - 将新的 React Native 架构引入开源社区
[*]React Conf 2024 - 第二天主旨演讲
新架构是什么
新架构是对支撑 React Native 的重要体系的全面重写,包括组件的渲染方式、JavaScript 抽象与原生抽象的通信方式以及跨不同线程的工作调理方式。尽管大多数用户无需相识这些体系的详细工作原理,但这些变革带来了性能提升和新的功能。
在旧架构中,React Native 使用异步桥接与原平生台通信。为了渲染一个组件或调用一个原生函数,React Native 必要将原生函数调用序列化并通过桥接队列化,这些调用将被异步处理。这种架构的利益是,主线程不会因渲染更新或处理原生模块函数调用而被阻塞,因为全部工作都在后台线程完成。
然而,用户盼望与交互的即时反馈,使应用能够像原生应用一样响应。这意味着某些更新必要在用户输入后同步渲染,大概会中断任何正在进行的渲染。由于旧架构仅支持异步,我们必要重写它以同时支持异步和同步的更新。
此外,在旧架构中,通过桥接序列化函数调用很快成为一个瓶颈,尤其是在频繁更新或处理大型对象时。这使得应用难以稳固地到达 60+ 帧每秒(FPS)。同时也存在同步题目:当 JavaScript 层和原生层不同步时,无法同步地调和它们,导致诸如列表显示空缺帧和因中心状态渲染导致的视觉 UI 跳跃等错误。
末了,由于旧架构使用原生条理结构保持单一的 UI 复制,并在原地修改该复制,布局只能在单一线程上计算。这使得处理诸如用户输入等紧急更新变得不大概,且布局无法同步读取,例如在布局效果中读取以更新工具提示的位置。
全部这些题目意味着无法准确支持 React 的并发特性。为相识决这些题目,新架构包括四个重要部分:
[*]新的原生模块体系
[*]新的渲染器
[*]事件循环
[*]移除桥接
新的原生模块体系允许 React Native 渲染器同步访问原生层,这使其能够异步和同步地处理事件、调理更新以及读取布局。新的原生模块默认采用懒加载,显著提升了应用的性能。
新的渲染器能够在多个线程上处理多个进行中的树,这使得 React 能够处理多个并发的更新优先级,无论是在主线程还是后台线程。它还支持同步或异步地从多个线程读取布局,以支持更响应的用户界面而不会出现卡顿。
新的事件循环能够按照明确定义的次序在 JavaScript 线程上处理使命。这使得 React 能够中断渲染以处理事件,从而使紧急的用户事件能够优先于较低优先级的 UI 过渡。事件循环还与 Web 规范保持一致,因此我们可以支持浏览器特性,如微使命(microtasks)、MutationObserver 和 IntersectionObserver。
末了,移除桥接允许更快的启动速率和 JavaScript 与原生运行时之间的直接通信,从而最小化工作切换的成本。这也使得错误报告和调试更加美满,减少了因未定义举动导致的崩溃。
新架构现已准备好用于生产环境。它已在 Meta 的 Facebook 应用和其他产品中大规模使用。我们在为 Quest 设备开发的 Facebook 和 Instagram 应用中成功使用了 React Native 和新架构。
我们的互助同伴已经在生产环境中使用新架构数月了:请查看 Expensify 和 Kraken 的成功案例,并实验 BlueSky 的新版本。
新的原生模块
新的原生模块体系是 JavaScript 与原平生台通信方式的庞大重写。它完全使用 C++ 编写,解锁了许多新功能:
[*]与原生运行时的同步访问
[*]JavaScript 与原生代码之间的范例安全
[*]跨平台的代码共享
[*]默认的懒加载模块
在新的原生模块体系中,JavaScript 与原生层如今可以通过 JavaScript 接口(JSI)进行同步通信,无需使用异步桥接。这意味着您的自定义原生模块如今可以同步调用一个函数,返回一个值,并将该值通报回另一个原生模块函数。
在旧架构中,为了处理来自原生函数调用的响应,您必要提供一个回调函数,并且返回的值必要是可序列化的:
// ❌ Sync callback from Native Module
nativeModule.getValue(value => {
// ❌ value cannot reference a native object
nativeModule.doSomething(value);
});
在新架构中,您可以对原生函数进行同步调用:
// ✅ Sync response from Native Module
const value = nativeModule.getValue();
// ✅ value can be a reference to a native object
nativeModule.doSomething(value);
有了新架构,您终于可以利用 C++ 原生实现的全部本领,同时仍旧可以通过 JavaScript/TypeScript API 访问它。新的模块体系支持用 C++ 编写的模块,因此您可以编写一次模块,它可以在全部平台上工作,包括 Android、iOS、Windows 和 macOS。用 C++ 实现模块允许更细粒度的内存管理和性能优化。
此外,借助 Codegen,您的模块可以在 JavaScript 层和原生层之间定义强范例的左券。根据我们的经验,跨边界的范例错误是跨平台应用程序崩溃最常见的原因之一。Codegen 让您能够降服这些题目,同时为您生成样板代码。
末了,模块如今是懒加载的:它们只有在现实必要时才加载到内存中,而不是在启动时加载。这减少了应用程序的启动时间,并在应用复杂性增加时保持较低的启动时间。
像 react-native-mmkv 这样的盛行库已经从迁移到新的原生模块中受益:
“新的原生模块极大地简化了 react-native-mmkv 的设置、主动链接和初始化。多亏了新架构,react-native-mkv 如今是一个纯 C++ 原生模块,这使它能够在任何平台上工作。新的 Codegen 使 MMKV 完全范例安全,通过强制 null 安全性,修复了恒久存在的 NullPointerReference 题目,并且能够同步调用原生模块函数,使我们能够用新的原生模块 API 替换自定义的 JSI 访问。”
新的渲染器
我们还完全重写了原生渲染器,带来了几个利益:
更新可以在不同的线程上以不同的优先级进行渲染。布局可以同步读取,并跨不同的线程进行。渲染器使用 C++ 编写,并在全部平台上共享。
更新后的原生渲染器如今将视图条理结构存储在不可变的树结构中。这意味着 UI 以一种无法直接更改的方式存储,允许线程安全地处理更新。这使得它能够处理多个进行中的树,每个树代表用户界面的不同版本。因此,更新可以在后台渲染而不会阻塞 UI(例如在过渡期间)或主线程(响应用户输入)。
通过支持多线程,React 可以中断低优先级的更新来渲染紧急更新,例如用户输入生成的更新,然后根据必要规复低优先级的更新。新的渲染器还可以同步地跨不同的线程读取布局信息。这使得低优先级的更新可以在后台计算,并在必要时进行同步读取,例如重新定位工具提示。
末了,用 C++ 重写渲染器使其能够在全部平台上共享。这确保了相同的代码可以在 iOS、Android、Windows、macOS 以及任何其他 React Native 支持的平台上运行,提供一致的渲染本领,而无需为每个平台重新实现。
这是实现我们多平台愿景的重要一步。例如,View Flattening 是一种仅用于 Android 的优化,以制止深条理的布局树。新的渲染器采用共享的 C++ 核心,将这一功能带到了 iOS。这种优化是主动的,不必要设置,随着共享渲染器一起免费提供。
通过这些更改,React Native 如今完全支持并发 React 功能,如 Suspense 和 Transitions,使构建复杂的用户界面变得更加轻易,这些界面能够快速响应用户输入而不会出现卡顿、延迟或视觉跳跃。未来,我们将利用这些新功能为内置组件如 FlatList 和 TextInput 带来更多改进。
像 Reanimated 这样盛行的库已经开始利用新的渲染器:
“Reanimated 4,目前正在开发中,引入了一个与新的渲染器直接协作的新动画引擎,允许其在不同线程上处理动画和管理布局。新的渲染器的设计真正使这些功能能够在不依赖众多变通方法的情况下构建。此外,由于它是用 C++ 实现并在全部平台上共享,Reanimated 的大部分代码可以一次编写,减少了平台特定的题目,最小化了代码库,并简化了对非树内平台的采用。”
事件循环
新架构使我们能够实现一个明确定义的事件循环处理模型,如本 RFC 中所述。该 RFC 依照 HTML 标准中描述的规范,并描述了 React Native 如何在 JavaScript 线程上执行使命。
实现一个明确定义的事件循环弥合了 React DOM 和 React Native 之间的差距:React Native 应用程序的举动如今更接近于 React DOM 应用程序的举动,使其更易于一次性学习和在任何地方编写。
事件循环为 React Native 带来了许多利益:
[*]中断渲染以处理事件和使命的本领
[*]与 Web 规范更细密的对齐
[*]为更多浏览器功能打下底子
借助事件循环,React 能够可猜测地排序更新和事件。这允许 React 用紧急的用户事件中断低优先级的更新,而新的渲染器使我们能够独立渲染这些更新。
事件循环还使事件和使命(如定时器)的举动与 Web 规范保持一致,这意味着 React Native 的工作方式更像用户在 Web 中熟悉的方式,并且允许 React DOM 和 React Native 之间更好的代码共享。
它还允许实现更多符合规范的浏览器功能,如微使命(microtasks)、MutationObserver 和 IntersectionObserver。这些功能在 React Native 中尚未准备好使用,但我们正在积极在未来为您提供这些功能。
末了,事件循环和新的渲染器的更改支持同步读取布局,使 React Native 能够为 useLayoutEffect 提供适当的支持,以同步读取布局信息并在同一帧中更新 UI。这使您能够在元素显示给用户之前准确定位它们。
有关更多详细信息,请参阅 useLayoutEffect。
移除桥接
在新架构中,我们还彻底移除了 React Native 对桥接的依赖,改为使用 JSI(JavaScript 接口)进行 JavaScript 和原生代码之间的直接、高效通信:
https://i-blog.csdnimg.cn/direct/6d3a04a862df49f3a8b152048e02dde3.png
移除桥接通过制止桥接初始化来提升启动时间。例如,在旧架构中,为了向 JavaScript 提供全局方法,我们必要在启动时在 JavaScript 中初始化一个模块,这会导致应用启动时间略有延迟:
// ❌ Slow initialization
import {NativeTimingModule} from 'NativeTimingModule';
global.setTimeout = timer => {
NativeTimingModule.setTimeout(timer);
};
// App.js
setTimeout(() => {}, 100);
在新架构中,我们可以直接从 C++ 绑定方法:
// ✅ Initialize directly in C++
runtime.global().setProperty(runtime, "setTimeout", createTimer);
// App.js
setTimeout(() => {}, 100);
重写还改善了错误报告,特殊是在启动时的 JavaScript 崩溃,并减少了因未定义举动导致的崩溃。假如发生崩溃,新的 React Native 开发工具简化了调试过程并支持新架构。为了向后兼容,桥接仍旧保留,以支持逐步迁移到新架构。未来,我们将完全移除桥接代码。
渐进式迁移
我们预计大多数应用可以以与其他版本相同的积极程度升级到 0.76。
当您升级到 0.76 时,新架构和 React 18 将默认启用。然而,为了使用并发特性并充实利用新架构的全部长处,您的应用和库必要逐步迁移以完全支持新架构。
在首次升级时,您的应用将运行在新架构上,并与旧架构通过主动互操作层进行交互。对于大多数应用来说,这无需任何更改即可正常工作,但互操作层存在已知的限制,因为它不支持访问自定义 Shadow Nodes 或并发特性。
要使用并发特性,应用还必要通过依照 React 规则来更新以支持并发 React。要将您的 JavaScript 代码迁移到 React 18 及其语义,请参阅 React 18 升级指南。
总体策略是让您的应用在不破坏现有代码的情况下运行在新架构上。然后,您可以以自己的节奏逐步迁移应用。对于已将全部模块迁移到新架构的新界面,您可以立刻开始使用并发特性。对于现有界面,您大概必要先解决一些题目并迁移模块,然后再添加并发特性。
我们已与最盛行的 React Native 库互助,以确保对新架构的支持。超过 850 个库已兼容新架构,包括全部每周下载量超过 20 万次的库(约占下载库的 10%)。您可以在 reactnative.directory 网站上检查库与新架构的兼容性:
https://i-blog.csdnimg.cn/direct/a23993c53d67415089fae2babee8caab.png
有关升级的更多详细信息,请参阅下方的《如何升级》。
新功能
新架构全面支持 React 18、并发特性以及 React Native 中的 useLayoutEffect。有关 React 18 功能的完整列表,请参阅 React 18 博客文章。
过渡(Transitions)
过渡是 React 18 中的一个新概念,用于区分紧急更新和非紧急更新。
紧急更新反映了直接的交互,如输入和点击。过渡更新则是将 UI 从一个视图过渡到另一个视图。
紧急更新必要立刻响应,以符合我们对物理对象举动的直觉。然而,过渡更新则不同,因为用户不盼望在屏幕上看到每一个中心值。在新架构中,React Native 能够分别支持渲染紧急更新和过渡更新。
通常,为了获得最佳的用户体验,单一的用户输入应导致一个紧急更新和一个非紧急更新。类似于 ReactDOM,诸如点击(press)或更改(change)等事件被视为紧急的并立刻渲染。您可以在输入事件中使用 startTransition API 来告知 React 哪些更新是“过渡”,并且可以延迟到后台处理:
import {startTransition} from 'react';
// Urgent: Show the slider value
setCount(input);
// Mark any state updates inside as transitions
startTransition(() => {
// Transition: Show the results
setNumberOfTiles(input);
});
将紧急事件与过渡分离,可以实现更具响应性的用户界面以及更直观的用户体验。
以下是没有过渡的旧架构(上)与有过渡的新架构(下)的对比:
https://i-blog.csdnimg.cn/direct/849f7bb0873c4ccfbaf37f08cb075783.gif
https://i-blog.csdnimg.cn/direct/f5aafad887b84cf0b6a420f00b5d85a9.gif
有关更多信息,请参阅《支持并发渲染器和功能》。
主动批处理
升级到新架构后,您将受益于 React 18 中的主动批处理。主动批处理允许 React 在渲染时将更多的状态更新批量处理,以制止渲染中心状态。这使得 React Native 更加快速,并且不易出现卡顿,而无需开发者编写任何额外代码。
https://i-blog.csdnimg.cn/direct/7095f3e3c1a94928859806b76c956f07.gif
https://i-blog.csdnimg.cn/direct/902753cf8d9f4e278ff2e40436410b7d.gif
在旧架构(左侧)中,会渲染更多的中心状态,即使滑块制止移动,UI 仍会连续更新。新架构(右侧)由于主动批处理更新,渲染的中心状态更少,渲染完成得更快。
有关更多信息,请参阅《支持并发渲染器和功能》。
useLayoutEffect
基于事件循环和同步读取布局的本领,在新架构中,我们为 React Native 增加了对 useLayoutEffect 的准确支持。
在旧架构中,您必要使用异步的 onLayout 事件来读取视图的布局信息(这也是异步的)。因此,至少会有一帧的布局是错误的,直到布局被读取和更新,这会导致工具提示等元素被放置在错误的位置,例如:
// ❌ async onLayout after commit
const onLayout = React.useCallback(event => {
// ❌ async callback to read layout
ref.current?.measureInWindow((x, y, width, height) => {
setPosition({x, y, width, height});
});
}, []);
// ...
<ViewWithTooltip
onLayout={onLayout}
ref={ref}
position={position}
/>;
新架构通过允许在 useLayoutEffect 中同步访问布局信息来解决这个题目:
// ✅ sync layout effect during commit
useLayoutEffect(() => {
// ✅ sync call to read layout
const rect = ref.current?.getBoundingClientRect();
setPosition(rect);
}, []);
// ...
<ViewWithTooltip ref={ref} position={position} />;
这个更改允许您同步读取布局信息,并在同一帧内更新 UI,从而在元素显示给用户之前准确定位它们:
https://i-blog.csdnimg.cn/direct/77f8b4fb61fa48aaa896e42f7498dff4.gif
https://i-blog.csdnimg.cn/direct/2e02685fcf144bbb9d0575d6c384dde7.gif
有关更多信息,请参阅《同步布局和效果》文档。
完整支持 Suspense
Suspense 允许您声明性地为组件树的一部分指定加载状态,当该部分尚未准备好显示时,可以显示一个备用内容:
<Suspense fallback={<Spinner />}>
<Comments />
</Suspense>
我们在几年前引入了有限版本的 Suspense,而 React 18 则增加了对其的完整支持。直到如今,React Native 尚未能够支持 Suspense 的并发渲染。
新架构包括 React 18 中引入的对 Suspense 的全面支持。这意味着您如今可以在 React Native 中使用 Suspense 来处理组件的加载状态,并且被挂起的内容将在后台渲染,而加载状态将被显示,从而为可见内容的用户输入赋予更高的优先级。
更多信息,请参阅 React 18 中的 Suspense RFC。
如何升级
要升级到 0.76,请按照发布公告中的步骤进行。由于此版本也升级到了 React 18,您还必要依照 React 18 升级指南。
这些步骤对于大多数应用来说已经足够,感谢新架构与旧架构之间的互操作层,可以顺遂升级到新架构。然而,为了充实利用新架构并开始使用并发特性,您必要迁移自定义的原生模块和原生组件,以支持新的原生模块和原生组件 API。
假如不迁移自定义的原生模块,您将无法享受到共享 C++、同步方法调用或来自 Codegen 的范例安全的上风。假如不迁移原生组件,您将无法使用并发特性。我们发起尽快将全部原生组件和原生模块迁移到新架构。
应用程序
假如您是应用开发者,为了完全支持新架构,您必要升级您的库、自定义原生组件和自定义原生模块,以全面支持新架构。
我们已经与最盛行的 React Native 库互助,确保对新架构的支持。您可以在 reactnative.directory 网站上检查库与新架构的兼容性。
假如您的应用依赖的任何库尚未兼容,您可以:
[*]在该库的堆栈中提交一个 issue,要求作者迁移到新架构。
[*]假如该库不再维护,考虑使用具有相同功能的替代库。
[*]在这些库迁移期间,选择不使用新架构。
假如您的应用有自定义的原生模块或自定义的原生组件,感谢我们的互操作层,我们预计它们可以正常工作。然而,我们发起将它们升级到新的原生模块和原生组件 API,以全面支持新架构并采用并发特性。
请按照以下指南将您的模块和组件迁移到新架构:
[*]原生模块
[*]原生组件
库
假如您是库的维护者,请起首测试您的库是否与互操作层兼容。假如不兼容,请在新架构工作组中提交一个 issue。
为了全面支持新架构,我们发起尽快将您的库迁移到新的原生模块(Native Module)和原生组件(Native Component)API。这将使您的库的用户能够充实利用新架构并支持并发特性。
您可以参考以下指南,将您的模块和组件迁移到新架构:
原生模块原生组件
选择退出
假如由于任何原因,新架构在您的应用中体现不正常,您始终可以选择退出,直到您准备好再次启用它。
要选择退出新架构:
在 Android 上,修改 android/gradle.properties 文件,并关闭 newArchEnabled 标志:
-newArchEnabled=true
+newArchEnabled=false
在 iOS 上,您可以通过运行以下下令重新安装依赖
RCT_NEW_ARCH_ENABLED=0 bundle exec pod install
感谢
将新架构交付给开源社区是一项巨大的积极,经过我们多年的研究和开发。我们想花一点时间感谢全部现任和曾经的 React 团队成员,他们帮助我们实现了这一成果。
我们也非常感谢全部与我们互助的互助同伴使这一切成为大概。详细来说,我们要特殊感谢:
[*]Expo,感谢其早期采用新架构,并支持迁移最盛行库的工作。
[*]Software Mansion,感谢其维护生态体系中的关键库,早期将其迁移到新架构,并在调查和修复各种题目上提供的帮助。
[*]Callstack,感谢其维护生态体系中的关键库,早期将其迁移到新架构,并支持社区 CLI 的工作。
[*]微软,感谢其为 react-native-windows 和 react-native-macos 以及其他多个开发工具添加新架构的实现。
[*]Expensify、Kraken、BlueSky 和 Brigad,感谢它们在采用新架构方面的先锋作用,并报告各种题目,使我们能够为其他人修复这些题目。
[*]全部独立的库维护者和开发者,感谢他们通过测试新架构、修复一些题目并就不明确的事项提出题目,为我们澄清题目作出的贡献。
免责声明:如果侵犯了您的权益,请联系站长,我们会及时删除侵权内容,谢谢合作!更多信息从访问主页:qidao123.com:ToB企服之家,中国第一个企服评测及商务社交产业平台。
页:
[1]