You signed in with another tab or window. Reload to refresh your session.You signed out in another tab or window. Reload to refresh your session.You switched accounts on another tab or window. Reload to refresh your session.Dismiss alert
安装,更新和卸载组件可能需要具有“副作用”。例如,React TODOs 应用程序可能需要在浏览器页面的标题中显示活动 TODO 项目的数量。直接使用 React API 是完成不了的。你需要使用 DOM API 。同样,在渲染输入表单时,你可能希望自动对焦文本框。这也必须使用 DOM API 完成。
十、Hooks
Hook 是一个特殊的函数,它可以让你“钩入” React 的特性。所有 Hooks 都以 use 开头。其中一些可为函数组件增加状态(如
useState
),一些可用于管理副作用(如useEffect),一些可用于缓存 memoize 函数和对象(如useCallback、 useMemo)。React hook 函数只能用于函数组件,不能在类组件中使用它们。
下面是一个基本示例:
1. 响应用户事件
我们给
button
组件增加一个onClick
事件:每次我们点击
Button
按钮时,onClick
都会调用内联箭头函数,向控制台输出Button clicked
。阅读和更新状态
跟踪状态更新并触发虚拟树协调算法更新到真实 DOM 上,React 需要了解组件中使用的所有元素发生的任何更改。为了更有效的执行此操作,React 需要为组件中引入的每个状态元素使用特殊的
getter
和setter
。这就是useState
发挥作用的地方。它定义了一个状态元素,并为它提供了一个getter
和setter
!useState
是允许你在 React 函数组件中添加 state 的 Hook。以下是我们尝试实现的 count state 元素所需的内容:
React.useState()
函数返回一个包含 2 个元素的数组。我们使用数组解构来命名,第一项是useState
返回的第一个值count
(getter),第二项是返回的第二个值setCount
函数(setter)。等价于下面的代码:第一项为定义的 state 变量名称,示例中叫
count
, 但是我们可以叫他任何名字,比如hello
。这是一种在函数调用时保存变量的方式 ——useState
是一种新方法,它与 class 里面的this.state
提供的功能完全相同。一般来说,在函数退出后变量就就会”消失”,而 state 中的变量会被 React 保留。第二项 "function" 将在调用时更改
state
元素的值(如果需要,它将触发DOM处理)。每次setCount
调用该函数时,React 都将重新渲染Button
组件,该组件将刷新组件中定义的所有变量(包括count
值)。我们传递给setCount
的参数将成为新的值count
。React.useState()
方法里面唯一的参数就是初始 state。不同于 class 的是,它可以是字符串,数字,数组等,而不一定是对象。在示例中,我们传了0
作为变量的初始state
。(如果我们想要在 state 中存储两个不同的变量,只需调用useState()
两次即可。)请注意我们不对 UI 本身进行任何操作,我们只是实现了一个
action
来改变 JavaScript 对象(在内存中)!React 负责将我们的声明性描述转换为浏览器中的实际UI。十一、多组件
将
Button
组件拆分为以下两个部分:Button
组件为按钮元素,并有一个静态标签;Display
组件以显示计数值。新
Display
组件将是纯粹的表示组件,没有自己的状态或交互。这很正常。并非每个 React 组件都必须与状态或交互挂钩。Display
组件的职责是简单地显示它将作为props
接收的值。例如,pre
元素用于托管值是该职责的一部分。该应用程序中的其他组件对此没有发言权!渲染兄弟组件
我们现在有两个要渲染的元素:
Button
和Display
。我们不能将它们直接相互渲染,如下所示:在React中,相邻的元素不能像这样呈现,因为当 JSX 被转换时,它们中的每一个都被转换为函数调用。有以下几个解决方案。
方案一:数组形式
当您渲染的所有元素都来自动态源时,这通常是一个很好的解决方案。但是,对于我们在这里做的情况,它并不理想。
方案二:添加父组件
为所有 React 元素添加一个共同的父组件。例如,我们可以将它们包含在
div
元素中。React API 支持此嵌套。事实上,你可以使用
React.Fragment
,它不会引入任何新的 DOM 父节点。方案三:使用 React.Fragment
方案三+:使用 <></>
你也可以这样写:
React 会将空标记转换为
React.Fragment
语法。但是,我们应该为 React 创建单一根节点,而不是我们刚刚执行的嵌套树。
顶级组件
让我们创建一个顶层组件来托管
Button
和Display
组件。现在的问题是:我们应该为这个新的父组件命名什么?由于这种新的父组件包含一个
Display
显示计数值,一个Button
增加计数值,我们可以把它命名为CountManager
。由于我们将在新
Display
组件中显示计数值,因此我们不再需要将计数值显示在按钮上。相反,我们可以在按钮上显示 “+1”。请注意,我还从
Button
组件中删除了state
元素,因为我们不能再使用它了。根据新要求,组件Button
和Display
组件都需要访问count
state 元素。当组件需要访问其兄弟组件所拥有的状态元素时,一种解决方案是将该状态元素“提升”到其最近共同父组件上。对于这种情况,父级是
CountManager
。通过将状态移动到
CountManager
,我们现在可以使用组件props
将数据从父级传到子级。通过
props
将count
值传递给Display
组件时,注意我使用了不同的名称(content
)。这很正常。你不必使用完全相同的名称。事实上,在某些情况下,引入新的通用名称对于子组件更好,因为它使状态可重用更高。由于
count
state元素现在位于CountManager
组件中,因此我们需要在CountManager
中处理更新它。我们来命名更新函数为incrementCounter
。该函数的逻辑实际上与我们之前在组件中的handleClick
函数中使用的逻辑相同。新incrementCounter
函数将更新CountManager
组件count
状态:为了使
Button
组件能够调用组件中的incrementCounter
函数,CountManager
将incrementCounter
作为props
参数给Button
组件。我们命名这个
props
为clickAction
,值为incrementCounter
,它是我们在CountManager
组件中定义的函数的引用。此
clickAction
属性允许Button
组件调用CountManager
组件的incrementCounter
功能。我们分析代码就会知道,
Button
组件是不知道单击它时会发生什么。它只遵循父级定义的规则并调用泛型clickAction
。父组件控制该行为的执行内容。这遵循责任隔离的概念。这里的每个组件都有一定的责任,它们专注于此。再看一下
Display
。从它的角度来看,计数值不是一个状态。它只是一个CountManager
组件传递给它的内容。Display
组件将始终显示该prop
。这也是职责分离。作为这些组件的设计者,你可以选择他们的职责级别。
我们创建
CountManager
组件负责管理计数状态。在项目开发中,又该如何做喃?我遵循的做法是在共享父节点中定义一个状态元素,该元素尽可能接近需要访问该状态元素的所有子节点。对于像这样的小应用程序,这通常意味着顶级组件本身。在较大的应用程序中,子树可以管理自己的状态“分支”,而不是依赖于在顶级根组件上定义的全局状态元素。
到目前为止,这是此示例的完整代码:
十二、组件可重用
组件都需要考虑可重用性。上栗中,
Button
组件也可以重用,它可以使用任何值增加count
计数,而不仅仅是+1
。首先
Button
在CountManager
组件中添加更多元素,以便我们可以测试这个新功能:Button
上面呈现的所有元素当前都有一个+1
标签,它们将使计数增加1。我们希望使它们显示特定于每个按钮的不同标签,并使它们根据特定于每个按钮的值执行不同的操作。请记住,你可以将任何值作为prop
传递给 React 元素。例如:
添加 props
我们需要做的第一件事是使组件中的
+1
标签Button
成为可自定义的标签。为了在 React 组件中进行自定义,我们引入了一个新的 prop(父组件可以控制)并使组件使用其值。在我们的例子中,我们可以让
Button
组件接收增量(1
,5
,10
),例如clickValue
。我们可以更改render
方法,CountManager
将我们想要测试的值传递给这个新的prop
。到目前为止,请注意有关此代码的一些事项:
count
。该Button
组件无需了解其click
事件的含义。它只需要触发click
事件时传递它。clickValue
属性的值(clickValue={5})
。我没有在那里使用字符串(clickValue="5")
。这是因为这里这里操作的是数字运算(每次Button
点击时),我需要这些值为数字。如果我将它们作为字符串传递,我将不得不在执行添加操作时将它转化为数字。自定义行为
在
CountManager
组件中需要做的另一件事是incrementCounter
功能。为了使函数通用,我们让它接收一个参数并使用该参数的值。例如:现在我们需要做的就是让
Button
组件使用clickValue
prop
作为其标签,并使其作为参数调用其onClick
事件clickValue
。使用内联箭头函数包装 onClick prop 以使其绑定到 Button 的
clickValue
。现在,三个按钮应以三个不同的点击值递增:
十三、接受用户的输入
想象一下,我们需要计算文本区域中用户类型的字符,就像 Twitter 的推文形式一样。对于每个字符的用户类型,我们需要使用新的字符数更新 UI 。
这是一个显示
textarea
输入元素的组件,其中包含字符数的占位符 div :要在用户输入时更新计数
textarea
,我们需要自定义用户键入时触发的事件。React 为此事件提供了onChange
方法。我们还需要使用 state 元素来计算字符数,并在onChange
事件中触发其 updater 函数。在
onChange
我们需要提出的新事件处理程序中,我们需要访问在textarea
元素中键入的文本。有两种主要方法来读取值。
document.getElementById
DOM API 来获取该元素,然后使用element.value
调用来读取它的值textarea
元素,我们可以通过 React ref 来获取 React 元素,然后访问值我们也可以
onChange
直接通过事件的目标对象访问元素。每个事件都暴露其目标,并且在目标上的onChange
事件textarea
是textarea
元素。这意味着我们需要做的就是:
这是最简单的解决方案,这个解决方案的不理想之处在于我们正在混淆问题。该
handleChange
事件具有调用setCount
函数和计算文本长度的副作用。我们需要混淆这些问题的原因是 React 不知道输入的是什么。这是一个 DOM 更新,而不是 React 更新。
我们可以通过覆盖其值
textarea
并通过 React 将其由状态更新变成 React 更新。在onChange
处理程序中,我们只设置在组件状态上键入的值,而不是对字符进行计数。以下是使用此策略的解决方案的一个版本:虽然这里代码量更多,但它有明确的关注点分离。React 现在知道并控制输入元素状态。此模式称为 React 中的受控组件模式。
此版本也更容易扩展。如果我们要计算用户输入的单词数量,这将成为另一个 UI 计算值。
十四、管理副作用
首次在浏览器中渲染 React 组件称为 “安装” ,将其从浏览器中删除称为“卸载”。
安装,更新和卸载组件可能需要具有“副作用”。例如,React TODOs 应用程序可能需要在浏览器页面的标题中显示活动 TODO 项目的数量。直接使用 React API 是完成不了的。你需要使用 DOM API 。同样,在渲染输入表单时,你可能希望自动对焦文本框。这也必须使用 DOM API 完成。
副作用通常需要在 React 的渲染任务之前或之后发生。这就是为什么 React 在类组件中提供“生命周期方法”以允许你在 render 方法之前或之后执行自定义操作的原因。你可以在组件首次安装在
componentDidMount
方法中后执行操作,你也可以在组件componentDidUpdate
方法中获取更新后执行操作,或者可以在componentWillUnmount
方法中删除之前执行操作。对于函数组件,使用
React.useEffect
hook 函数管理副作用,该函数有2个参数:回调函数和依赖项数组。第一次 React 呈现一个有
useEffect
调用的组件时,它将调用它的回调函数。在每个新组件呈现之后,如果依赖项的值与之前渲染中的值不同,则 React 将再次调用回调函数。副作用方法对于分析应用程序中正在发生的事情以及进一步优化 React 的性能也非常方便。
The text was updated successfully, but these errors were encountered: