关注黑人生命。 支持司法公正。

React v17.0 RC 版本发布:没有新特性

August 10, 2020 by Dan Abramov and Rachel Nabors

今天,我们发布了 React 17 的第一个 RC 版本。自 React 上一个主要版本至今已经两年半了,按照我们的标准,时间跨度有些长了!在此篇博客中,我们将讲解此次主要版本的职责,对你的影响以及如何试用此版本。

无新特性

React 17 的版本是非比寻常的,因为它没有添加任何面向开发人员的新功能。而主要侧重于升级简化 React 本身

我们正在积极开发 React 的新功能,但它们并不属于此版本。React 17 是我们进行深度推广战略的关键所在。

此版本之所以特殊,你可以认为React 17 是 “垫脚石” 版本,它会使得由一个 React 版本管理的 tree 嵌入到另一个 React 版本管理的 tree 中时会更加安全。

逐步升级

在过去 7 年里,React 一直遵循 “all-or-nothing” 的升级策略。你可以继续使用旧版本,也可以将整个应用程序升级至新版本。但没有介于两者之间的情况。

此方式持续至今,但是我们遇到了 “all-or-nothing” 升级策略的局限性。许多 API 的变更,例如,弃用旧版 context API 时,并不能以自动化的方式来完成。至今可能大多数应用程序从未使用过它们,但我们仍然选择在 React 中支持它们。我们必须在无限期支持过时的 API 或针对某些应用仍使用旧版本 React 间进行选择。但这两个方案都不合适。

因此,我们想提供另一种方案。

React 17 开始支持逐步升级 React 版本。当从 React 15 升级至 16 时(或者从 React 16 升级至 17时),通常会一次升级整个应用程序。这适用于大部分应用程序。但是,如果代码库是在几年前编写的,并且并没有得到很好的维护,那么升级它会变得越来越有挑战性。尽管可以在页面上使用两个版本的 React,但是直到 React 17 依旧有事件问题出现。

我们使用 React 17 解决了许多诸如此类的问题。这将意味着当 React 18 或未来版本问世时,你将有更多选择。首选还是像以前一样,一次升级整个应用程序。但你也可以选择逐步升级你的应用程序。例如,你可能会将大部分应用程序迁移至 React 18,但在 React 17 上保留一些延迟加载的对话框或子路由。

但这不意味着你必须逐步升级。对于大部分应用程序来说,一次全量升级仍是最好的解决方案。加载两个 React 版本,即使其中一个是按需延迟加载的,仍然不太理想。但是,对于没有积极维护的大型应用来说,可以考虑此种方案,并且 React 17 开始可以保证这些应用程序不落伍。

为了实现逐步升级,我们需要对 React 事件系统进行一些更改。而这些更改可能会对代码产生影响,这也是 React 17 成为主要版本的原因。实际上,10 万个以上的组件中受影响的组件不超过 20 个,因此,我们希望大多数应用程序都可以升级到 React 17,而不会产生太多影响。如果你遇到问题,请联系我们

逐步升级 Demo

我们准备了一个示例 repo,展示了如何在必要时延迟加载旧版本的 React。此 demo 使用了 Create React App 进行构建,但对其他工具采用类似的方法应该也适用。我们欢迎使用其他工具的开发者编写 demo 并提交 PR。

注意

我们已将其他更改推迟到 React 17 之后。此版本的目标是实现逐步升级。如果升级 React 17 太困难,则此目标会无法实现。

更改事件委托

从技术上讲,始终可以在应用程序中嵌套不同版本的 React。但是,由于 React 事件系统的工作原理,这很难实现。

在 React 组件中,通常会内联编写事件处理:

<button onClick={handleClick}>

与此代码等效的原生 DOM 操作如下:

myButton.addEventListener('click', handleClick);

但是,对大多数事件来说,React 实际上并不会将它们附加到 DOM 节点上。相反,React 会直接在 document 节点上为每种事件类型附加一个处理器。这被称为事件委托。除了在大型应用程序上具有性能优势外,它还使添加类似于 replaying events 这样的新特性变得更加容易。

自从其发布以来,React 一直自动进行事件委托。当 document 上触发 DOM 事件时,React 会找出调用的组件,然后 React 事件会在组件中向上 “冒泡”。但实际上,原生事件已经冒泡出了 document 级别,React 在其中安装了事件处理器。

但是,这就是逐步升级的困难所在。

如果页面上有多个 React 版本,他们都将在顶层注册事件处理器。这会破坏 e.stopPropagation():如果嵌套树结构中阻止了事件冒泡,但外部树依然能接收到它。这会使不同版本 React 嵌套变得困难重重。这种担忧并不是没有根据的 —— 例如,四年前 Atom 编辑器就遇到了相同的问题

这也是我们为什么要改变 React 底层附加事件方式的原因。

在 React 17 中,React 将不再向 document 附加事件处理器。而会将事件处理器附加到渲染 React 树的根 DOM 容器中:

const rootNode = document.getElementById('root');
ReactDOM.render(<App />, rootNode);

在 React 16 或更早版本中,React 会对大多数事件执行 document.addEventListener()。React 17 将会在底层调用 rootNode.addEventListener()

A diagram showing how React 17 attaches events to the roots rather than to the document

由于此更改,现在可以更加安全地进行新旧版本 React 树的嵌套。请注意,要使其正常工作,两个版本都必须为 17 或更高版本,这就是为什么强烈建议升级到 React 17 的根本原因。从某种意义上讲,React 17 是一个 “垫脚石” 版本,使逐步升级成为可能。

此更改还使得将 React 嵌入使用其他技术构建的应用程序变得更加容易。例如,如果应用程序的“外壳”是用 jQuery 编写的,但其中较新的代码是用 React 编写的,则 React 代码中的 e.stopPropagation() 会阻止它影响 jQuery 的代码 —— 这符合预期。换个角度来说,如果你不再喜欢 React 并想重写应用程序(比如,用 jQuery),则可以从外壳开始将 React 转换为 jQuery,而不会破坏事件冒泡。

经核实,多年来在 issue 追踪器报告许多问题新特性解决,这些问题大多都与将 React 与非 React 代码集成有关。

注意

你可能想知道这是否会破坏根 DOM 容器之外的 Portals。答案是 React 还会监听 portals 容器上的事件,所以这不是问题。

解决隐患

与其他重大更改一样,可能需要对代码进行调整。在 Facebook,我们在成千上万个模块中,大约调整了 10 个模块以适应此更改。

例如,如果模块中使用 document.addEventListener(...) 手动添加了 DOM 监听,你可能希望能捕获到所有 React 事件。在 React 16 或更早版本中,即使你在 React 事件处理器中调用 e.stopPropagation(),你创建的 DOM 监听仍会触发,这是因为原生事件已经处于 document 级别。使用 React 17 冒泡将被阻止(按需),因此你的 document 级别的事件监听不会触发:

document.addEventListener('click', function() {
  // This custom handler will no longer receive clicks
  // from React components that called e.stopPropagation()
});

你可以将监听转换为使用捕获来修复此类代码。为此,你可以将 { capture: true } 作为 document.addEventListener 的第三个参数传递:

document.addEventListener('click', function() {
  // Now this event handler uses the capture phase,
  // so it receives *all* click events below!
}, { capture: true });

请注意,此策略在全局上具有更好的适应性。例如,它可能会修复代码中现有的错误,这些错误在 React 事件处理器外部调用 e.stopPropagation() 发生。换句话说,React 17 的事件冒泡更接近常规 DOM

其他重大更改

我们将 React 17 中的重大更改保持在最低水平。例如,它不会删除以前版本中弃用的任务方法。但是,它的确包含一些其他重大更改,根据经验,这些更改会相对安全。总体而言,由于这些因素的存在,在 10 万个以上的组件中受影响的组件不超过 20 个。

对标浏览器

我们对事件系统进行了一些较小的更改:

  • onScroll 事件不再冒泡,以防止出现常见的混淆
  • React 的 onFocusonBlur 事件已在底层切换为原生的 focusinfocusout 事件。它们更接近 React 现有行为,有时还会提供额外的信息。
  • 捕获事件(例如,onClickCapture)现在使用的是实际浏览器中的捕获监听器。

这些更改会使 React 与浏览器行为更接近,并提高了互操作性。

注意:

尽管 React 17 底层已将 onFocus 事件从 focus 切换为 focusin,但请注意,这并未影响冒泡行为。在 React 中,onFocus 事件总是冒泡的,在 React 17 中会继续保持,因为通常它是一个更有用的默认值。请参阅 sandbox,以了解为不同特定用例添加不同检查。

去除事件池

React 17 中移除了 “event pooling(事件池)“。它并不会提高现代浏览器的性能,甚至还会使经验丰富的开发者一头雾水:

function handleChange(e) {
  setData(data => ({
    ...data,
    // This crashes in React 16 and earlier:
    text: e.target.value
  }));
}

这是因为 React 在旧浏览器中重用了不同事件的事件对象,以提高性能,并将所有事件字段在它们之前设置为 null。在 React 16 及更早版本中,使用者必须调用 e.persist() 才能正确的使用该事件,或者正确读取需要的属性。

在 React 17 中,此代码可以按照预期效果执行。旧的事件池优化操作已被完成删除,因此,使用者可以在需要时读取事件字段。

这改变了行为,因此我们将其标记为重大更改,但在实践中我们没有看到它在 Facebook 上造成影响。(甚至还修复了一些错误!)请注意,e.persist() 在 React 事件对象中仍然可用,只是无效果罢了。

副作用清理时间

我们将使 useEffect 和清理函数的时间保持一致。

useEffect(() => {
  // This is the effect itself.
  return () => {    // This is its cleanup.  };});

大多数副作用(effect)不需要延迟屏幕更新,因此 React 在屏幕上反映出更新后立即异步执行它们。(在极少数情况下,你需要一种副作用来阻止绘制,例如,如果需要获取尺寸和位置,请使用 useLayoutEffect。)

然而,副作用清理函数(如果存在)在 React 16 中同步运行。我们发现,对于大型应用程序来说,这不是理想选择,因为同步会减缓屏幕的过渡(例如,切换标签)。

在 React 17 中,副作用清理函数会异步执行 —— 如果要卸载组件,则清理会在屏幕更新运行。

这反映了副作用本身如何更紧密地运行。在极少数情况下,你可能希望依靠同步执行,可以改用 useLayoutEffect

注意

你可能想知道这是否意味着你现在将无法修复有关未挂载组件上的 setState 的警告。不必担心,React 专门处理了这种情况,并且不会在卸载和清理之间短暂间隔内发出 setState 的警告。因此,取消代码的请求或间隔几乎总是可以保存不变的。

此外,React 17 会根据它们在树中的位置,以与效果相同的顺序执行清除功能。以前,顺序有时会不同。

隐患

可复用的库可能需要对此情况进行深度测试,但我们只遇到了几个组件会因为这次改变出现问题。有问题的代码的其中一个示例如下所示:

useEffect(() => {
  someRef.current.someSetupMethod();
  return () => {
    someRef.current.someCleanupMethod();
  };
});

问题在于 someRef.current 是可变的,因此在运行清除函数时,它可能已经设置为 null。解决方案是在副作用内部存储会发生变化的值:

useEffect(() => {
  const instance = someRef.current;
  instance.someSetupMethod();
  return () => {
    instance.someCleanupMethod();
  };
});

我们不希望此问题对大家造成影响,我们提供了 eslint-plugin-react-hooks/exhaustive-deps 的 lint 规则(请确保在项目中使用它)会对此情况发出警告。

返回一致的 undefined 错误

在 React 16 及更早版本中,返回 undefined 始终是一个错误:

function Button() {
  return; // Error: Nothing was returned from render
}

部分原因是这很容易无意间返回 undefined

function Button() {
  // We forgot to write return, so this component returns undefined.
  // React surfaces this as an error instead of ignoring it.
  <button />;
}

以前,React 只对 class 和函数组件执行此操作,但并不会检查 forwardRefmemo 组件的返回值。这是由于编码错误导致。

在 React 17 中,forwardRefmemo 组件的行为会与常规函数组件和 class 组件保持一致。在返回 undefined 时会报错

let Button = forwardRef(() => {
  // We forgot to write return, so this component returns undefined.
  // React 17 surfaces this as an error instead of ignoring it.
  <button />;
});

let Button = memo(() => {
  // We forgot to write return, so this component returns undefined.
  // React 17 surfaces this as an error instead of ignoring it.
  <button />;
});

对于不想进行任何渲染的情况,请返回 null

原生组件栈

当你在浏览器中遇到错误时,浏览器会为你提供带有 JavaScript 函数的名称及位置的调用栈信息。然而,JavaScript 调用栈通常不足以诊断问题,因为 React 树的层次结构可能同样重要。你不仅要知道哪个 Button 抛出了错误,而且还想知道 Button 在 React 树中的哪个位置。

为了解决这个问题,当你遇到错误时,从 React 16 开始会打印 “组件栈” 信息。尽管如此,它们仍然不如原生的 JavaScript 调用栈。特别是,它们在控制台中不可点击,因为 React 不知道函数在源代码中的声明位置。此外,它们在生产中几乎无用。不同于常规压缩后的 JavaScript 调用栈,它们可以通过 sourcemap 的形式自动恢复到原始函数的位置,而使用 React 组件栈,在生产环境下必须在调用栈信息和 bundle 大小间进行选择。

在 React 17 中,使用了不同的机制生成组件调用栈,该机制会将它们与常规的原生 JavaScript 调用栈缝合在一起。这使得你可以在生产环境中获得完全符号化的 React 组件调用栈信息。

React 实现这一点的方式有点非常规。目前,浏览器无法提供获取函数调用栈框架(源文件和位置)的方法。因此,当 React 捕获到错误时,将通过组件上述组件内部抛出的临时错误(并捕获)来重建其组件调用栈信息。这会增加崩溃时的性能损失,但每个组件类型只会发生一次。

如果你对此感兴趣,可以在这个 PR 中阅读更多详细信息,但是在大多数情况下,这种机制不会影响你的代码。从使用者的角度来看,新功能就是可以单击组件调用栈(因为它们依赖于本机浏览器调用栈框架),并且可以像常规 JavaScript 错误那样在生产中进行解码。

构成重大变化的部分是,要使此功能正常工作,React 将在捕获错误后在调用栈中重新执行上面某些函数和某些 class 构造函数的部分。由于渲染函数和 class 构造函数不应具有副作用(这对于 SSR 也很重要),因此这不会造成任何实际问题。

移除私有导出

最后,值得注意的重大变化时我们删除了一些以前暴露给其他项目的 React 内部组件。特别是,React Native for Web 过去常常依赖于事件系统的某些内部组件,但这种依赖关系很脆弱且经常被破坏。

在 React 17 中,这些私有导出已被移除。据我们所知,React Native for Web 是唯一使用它们的项目,它们已经完成了向不依赖那些私有导出函数的其他方法迁移。

这意味着旧版本的 React Native for Web 不会与 React 17 兼容,但是新版本可以使用它。实际上,并没有太大的变化,因为 React Native for Web 必须发布新版本以适应其内部 React 的变化。

另外,我们删除了 ReactTestUtils.SimulateNative 的 helper 方法。他们从未被记录,没有按照他们名字所暗示的那样去做,也没有处理我们对事件系统所做的更改。如果你想要一种简便的方式来触发测试中原生浏览器的事件,请改用 React Testing Library

安装

我们鼓励你尽快尝试 React 17.0 RC 版本,在迁移过程中遇到任何问题都可以向我们提出请注意,候选版本没有稳定版本稳定,因此请不要将其部署到生产环境。

通过 npm 安装 React 17 RC 版,请执行:

npm install react@17.0.0-rc.1 react-dom@17.0.0-rc.1

通过 yarn 安装 React 17 RC 版,请执行:

yarn add react@17.0.0-rc.1 react-dom@17.0.0-rc.1

我们还通过 CDN 提供了 React RC 的 UMD 构建版本:

<script crossorigin src="https://unpkg.com/react@17.0.0-rc.1/umd/react.production.min.js"></script>
<script crossorigin src="https://unpkg.com/react-dom@17.0.0-rc.1/umd/react-dom.production.min.js"></script>

有关详细安装说明,请参阅文档。

Changelog

React

  • 为全新的 JSX 转换器添加 react/jsx-runtimereact/jsx-dev-runtime。(@lunaruan 提交于 #18299
  • 根据原生框架构建组件调用栈。(@sebmarkbage 提交于 #18561
  • 可以在 context 中设置 displayName 以改善调用栈信息。(@eps1lon 提交于 #18224
  • 防止 'use strict' 从 UMD 的 bundles 中泄露。(@koba04 提交于 #19614
  • 停止使用 fb.me 进行重定向。(@cylim 提交于 #19598

React DOM

  • 将事件委托从 document 切换为 root。(@trueadm 提交于 #18195其他
  • 在运行下一个副作用前,请清理所有副作用。(@bvaughn 提交于 #17947
  • 异步运行 useEffect 清理函数。(@bvaughn 提交于 #17925
  • 使用浏览器的 focusinfocusout 替换 onFocusonBlur 的底层实现。(@trueadm 提交于 #19186
  • 将所有 Capture 事件都使用浏览器的捕获阶段实现。(@trueadm 提交于 #19221
  • 禁止在 onScroll 事件时冒泡。(@gaearon 提交于 #19464)
  • 如果 forwardRefmemo 组件的返回值为 undefined,则抛出异常。(@gaearon 提交于 #19550
  • 移除事件池。(@trueadm 提交于 #18969
  • 移除 React Native Web 不需要的内部组件。(@necolas 提交于 #18483)
  • 当挂载 root 时,附加所有已知的事件监听器。(@gaearon 提交于 #19659
  • 在 Dev 模式下,禁用第二次渲染过程中的 console。(@sebmarkbage 提交于 #18547
  • 弃用为记录且具有误导性的 ReactTestUtils.SimulateNative API。(@gaearon 提交于 #13407)
  • 重命名内部使用的私有字段(@gaearon 提交于 #18377
  • 不在开发环境调用 User Timing API。(@gaearon 提交于 #18417
  • 在严格模式下重复渲染期间禁用 console。(@sebmarkbage 提交于 #18547
  • 在严格模式下,二次渲染组件也不使用 Hook。(@eps1lon 提交于 #18430
  • 允许在生命周期函数中调用 ReactDOM.flushSync(但会发出警告)。(@sebmarkbage 提交于 #18759
  • code 属性添加到键盘事件对象中。(@bl00mber 提交于 #18287
  • video 元素添加 disableRemotePlayback 属性。(@tombrowndev 提交于 #18619
  • input 元素添加 enterKeyHint 属性。(@eps1lon 提交于 #18634
  • 当没有给 <Context.Provider> 提供任何值时,会发出警告。(@charlie1404 提交于 #19054
  • 如果 forwardRefmemo 组件的返回值为 undefined,则抛出警告。(@bvaughn 提交于 #19550
  • 为无效更新改进错误信息。(@JoviDeCroock 提交于 #18316
  • 从调用栈信息中忽略 forwardRef 和 memo。(@sebmarkbage 提交于 #18559
  • 在受控输入与非受控输入间切换时,改善错误消息。(@vcarl 提交于 #17070
  • 保持 onTouchStartonTouchMoveonWheel 默认为 passive。(@gaearon 提交于 #19654
  • 修复在 development 模式下 iframe 关闭时,setState 挂起的问题。(@gaearon 提交于 #19220
  • 使用 defaultProps 修复拉架子组件在渲染时的问题。(@jddxf 提交于 #18539
  • 修复当 dangerouslySetInnerHTMLundefined 时,误报警告的问题。(@eps1lon 提交于 #18676
  • 使用费标准的 require 实现来修复 Test Utils。(@just-boris 提交于 #18632
  • 修复 onBeforeInput 报告错误的 event.type。(@eps1lon 提交于 #19561
  • 修复 Firefox 中 event.relatedTarget 输出为 undefined 的问题。(@claytercek 提交于 #19607
  • 修复 IE11 中 “unspecified error” 的问题。(@hemakshis 提交于 #19664
  • 修复 shadow root 中的渲染问题。(@Jack-Works 提交于 #15894
  • 使用事件捕获修复 movementX/Y polyfill 的问题。(@gaearon 提交于 #19672
  • 使用委托处理 onSubmitonReset 事件。(@gaearon 提交于 #19333
  • 提高内存使用率。(@trueadm 提交于 #18970

React DOM Server

  • 使用服务端渲染的 useCallbackuseMemo 一致。(@alexmckenley提交于 #18783
  • 修复函数组件抛出异常时状态泄露的问题。(@pmaccart 提交于 #19212

React Test Renderer

Concurrent Mode(实验阶段)

  • 改进启发式更新算法。(@acdlite 提交于 #18796
  • 在实现性 API 前添加 unstable_ 前缀。 (@acdlite 提交于 #18825)
  • 移除 unstable_discreteUpdatesunstable_flushDiscreteUpdates。(@trueadm 提交于 #18825
  • 移除了 timeoutMs 参数。(@acdlite 提交于 #19703
  • 禁用 <div hidden /> 预渲染,以支持未来的 API。(@acdlite 提交于 #18917
  • 添加了一个实现性的 unstable_useOpaqueIdentifier Hook。(@lunaruan 提交于 #17322
  • 添加了一个实验性的 unstable_startTransition API. (@rickhanlonii 提交于 #19696)
  • 在测试渲染器中使用 act 后,不在刷新 Suspense 的 fallback。(@acdlite 提交于 #18596
  • 将全局渲染的 timeout 用于 CPU Suspense。(@sebmarkbage 提交于 #19643
  • 挂载前,清除现有根目录的内容。(@bvaughn 提交于 #18730
  • 修复带有错误边界的 bug。(@acdlite 提交于 #18265
  • 修复了导致挂起树更新丢失的 bug。(@acdlite 提交于 #18384 以及 #18457
  • 修复导致渲染阶段更新丢失的 bug。(@acdlite 提交于 #18537
  • 修复 SuspenseList 的 bug。(@sebmarkbage 提交于 #18412
  • 修复导致 Suspense fallback 过早显示的 bug。(@acdlite 提交于 #18411
  • 修复 SuspenseList 中使用 class 组件异常的 bug。(@sebmarkbage 提交于 #18448
  • 修复输入内容可能被更新被丢弃的 bug。(@jddxf 提交于 #18515 以及 @acdlite 提交于 #18535
  • 修复暂挂 Suspense fallback 后卡住的错误。(@acdlite 提交于 #18663
  • 如果 hydrate 中,不要切断 SuspenseList 的尾部。(@sebmarkbage 提交于 #18854
  • 修复 useMutableSource 中的 bug,此 bug 可能在 getSnapshot 更改时出现。(@bvaughn 提交于 #18297
  • 修复 useMutableSource 令人恶心的 bug。(@bvaughn 提交于 #18912
  • 如果外部渲染且提交之前调用 setState,会发出警告。(@sebmarkbage 提交于 #18838
Is this page useful?编辑此页面