React函数组件中与生命周期相关Hooks详解

embedded/2025/1/11 18:34:14/

React 函数组件及其钩子渲染流程是 React 框架中的核心概念之一。以下是对该流程的详细解析:

一、React 函数组件基础

  1. 定义

    React 函数组件是一个接收 props 作为参数并返回 React 元素的函数。它们通常用于表示 UI 的一部分,并且不保留内部状态(除非使用 React 的 Hooks)。

  2. 特点

    • 简洁明了,易于理解和维护。
    • 适用于表示无状态或简单状态的 UI 组件。
    • 可以使用 Hooks 来添加状态和其他 React 功能。

二、React Hooks 渲染流程

React Hooks 允许你在函数组件中使用状态和其他 React 功能。以下是对 React Hooks 渲染流程的详细解析:

  1. 初始渲染

    • 当 React 应用启动时,会创建根组件的实例,并调用函数组件来生成虚拟 DOM 树。
    • 在函数组件中,如果使用了 Hooks(如 useStateuseEffect 等),React 会按照 Hooks 的调用顺序将它们保存在一个内部数组中。
    • 对于 useState Hook,它会初始化状态并返回一个包含当前状态和更新函数的数组。
    • 对于 useEffect Hook,它会在首次渲染后执行(相当于类组件的 componentDidMount),并且会在依赖项发生变化时重新执行
  2. 更新流程

    • 当组件的状态通过 setState 方法更新,或者父组件传递的属性(props)发生变化时,组件会进入更新流程。
    • React 会再次调用函数组件来生成新的虚拟 DOM 树。
    • 在更新过程中,React 会按照 Hooks 的调用顺序重新调用它们,并更新它们的状态或执行副作用。
    • 如果 useEffect Hook 的依赖项发生了变化,它会重新执行。
  3. 渲染优化

    • React 采用了一些优化策略来提高渲染性能,如避免不必要的重新渲染。
    • useMemouseCallback Hooks 可以用于缓存计算结果和避免不必要的函数重新创建。
    • React.memo 可以用于优化函数组件的重新渲染,只有当 props 发生变化时才重新渲染组件。
  4. 错误处理

    • React 提供了错误边界组件来捕获组件渲染过程中的错误。
    • 错误边界组件可以捕获子组件中的错误,并显示备用 UI,而不是让整个应用崩溃。

三、useEffect 钩子详解

  1. 作用

    • useEffect Hook 用于在函数组件中执行副作用操作,如数据获取、订阅事件、手动修改 DOM 等。
    • 它会在组件首次渲染后以及后续每次更新后执行(除非依赖项没有变化)。
  2. 参数

    • useEffect 接受两个参数:一个回调函数和一个依赖项数组。
    • 回调函数包含要执行的副作用操作。
    • 依赖项数组用于指定何时重新执行回调函数。当数组中的值发生变化时,回调函数会重新执行。
  3. 清理

    • 回调函数可以返回一个清理函数,该函数会在组件卸载或下次运行 useEffect 之前执行。
    • 清理函数用于取消订阅、清理计时器等操作,以避免内存泄漏或不必要的副作用。

综上所述,React 函数组件及其钩子渲染流程是 React 应用中的关键部分。通过合理使用函数组件和 Hooks,可以构建高效、可维护的 React 应用。
是的,在 React 中,useEffect 钩子函数通常会在组件的 DOM 更新完成后再执行。useEffect 可以看作是 componentDidMountcomponentDidUpdatecomponentWillUnmount 这三个类组件生命周期方法的组合。

具体来说:

  1. 首次渲染后执行:当组件首次渲染到屏幕上,React 会将 DOM 更新完成后调用 useEffect。这相当于类组件中的 componentDidMount

  2. 后续更新后执行:在后续的渲染中(即当组件的 props 或 state 发生变化导致重新渲染时),React 依然会在 DOM 更新完成后调用 useEffect。这相当于类组件中的 componentDidUpdate

  3. 清理副作用useEffect 还可以返回一个清理函数,该函数会在组件卸载或下次运行 useEffect 之前执行,这相当于类组件中的 componentWillUnmount

以下是一个简单的例子,展示了 useEffect 的用法:

import React, { useState, useEffect } from 'react';function ExampleComponent() {const [count, setCount] = useState(0);useEffect(() => {// 这里的代码会在组件首次渲染后以及后续每次更新后执行console.log('DOM 更新完成,useEffect 被调用');// 返回一个清理函数,该函数会在组件卸载或下次运行 useEffect 之前执行return () => {console.log('清理副作用');};}, [count]); // 注意:第二个参数是依赖数组,只有当数组中的值发生变化时,useEffect 才会重新执行return (<div><p>You clicked {count} times</p><button onClick={() => setCount(count + 1)}>Click me</button></div>);
}export default ExampleComponent;

在这个例子中,每当 count 发生变化时,useEffect 都会执行,并且在组件卸载时(例如,当组件被移除或替换时),清理函数会执行。

需要注意的是,如果 useEffect 的依赖数组为空([]),则 useEffect 只在组件首次渲染和卸载时执行一次,这类似于 componentDidMountcomponentWillUnmount 的组合。

在 React 中,函数组件本身并不具有传统类组件那样的生命周期方法,如 componentDidMountshouldComponentUpdatecomponentWillUnmount 等。然而,从 React 16.8 版本开始,React 引入了 Hooks API,这使得函数组件能够使用类似类组件生命周期的功能。

四、与生命周期相关Hooks

以下是与生命周期相关的函数组件钩子及其作用:

  1. useEffect

    • 作用:用于在函数组件中执行副作用操作,这些操作可能包括数据获取、订阅事件、手动修改 DOM 等。
    • 与生命周期的关系useEffect 可以模拟 componentDidMountcomponentDidUpdate 这两个生命周期方法。当组件首次渲染后,以及后续每次更新后(依赖项发生变化时),useEffect 中的回调函数都会执行。此外,useEffect 还可以返回一个清理函数,该函数会在组件卸载或下次运行 useEffect 之前执行,模拟 componentWillUnmount
  2. useLayoutEffect

    • 作用:与 useEffect 类似,但 useLayoutEffect 中的回调函数会在所有的 DOM 变更之后同步调用,可以用于读取 DOM 布局并同步触发重渲染。在浏览器执行绘制之前,useLayoutEffect 内部的更新计划将被同步刷新。
    • 与生命周期的关系:由于 useLayoutEffect 在 DOM 变更后同步调用,因此它更接近于类组件中的 componentDidMountcomponentDidUpdate,但执行时机略有不同。
    • useLayoutEffect 是 React 中的一个 Hook,用于在浏览器布局和绘制之前同步执行副作用。以下是关于 useLayoutEffect 的详细解析:

一、作用与特点

  1. 同步执行

    • useLayoutEffect 中的回调函数会在所有的 DOM 变更之后同步调用,即在浏览器执行绘制之前执行。
    • 这意味着 useLayoutEffect 内部的更新计划会被同步刷新,从而允许在绘制之前对 DOM 进行必要的调整。
  2. useEffect 的区别

    • useEffect 是在浏览器绘制完成后异步执行副作用,而 useLayoutEffect 则是在绘制之前同步执行。
    • 因此,useLayoutEffect 更适合用于需要在 DOM 更新之前进行一些计算或修改 DOM 的场景。
  3. 性能考虑

    • 由于 useLayoutEffect 是同步执行的,如果其执行时间过长,可能会阻塞页面渲染,导致用户看到延迟。
    • 因此,在大多数情况下,应优先使用 useEffect,只有在需要同步执行副作用时才考虑使用 useLayoutEffect

二、使用场景

  1. 读取 DOM 布局

    • useLayoutEffect 可以在 DOM 更新后立即读取布局信息,如元素的位置、尺寸等,并据此进行同步调整。
  2. 防止闪屏

    • 在某些情况下,使用 useEffect 可能会导致视图元素的位置或大小发生变化,从而产生闪屏效果。
    • 使用 useLayoutEffect 可以在浏览器绘制之前计算好元素的位置和大小,从而避免闪屏问题。
  3. 集成非 React DOM 库

    • 当需要与非 React DOM 库集成时,可能需要在 DOM 更新后立即执行一些操作。
    • useLayoutEffect 提供了在绘制之前执行这些操作的机会。
      三、写法与示例
  4. 基本写法

    useLayoutEffect(() => {// 执行副作用操作return () => {// 清理函数,组件卸载时执行};
    }, [dependencies]); // 依赖项数组,可选
    
  5. 示例
    假设有一个场景,需要在组件渲染后立即将一个元素的宽度设置为窗口宽度的一半。可以使用 useLayoutEffect 来实现:

    import React, { useRef, useLayoutEffect } from 'react';function MyComponent() {const elementRef = useRef(null);useLayoutEffect(() => {if (elementRef.current) {elementRef.current.style.width = window.innerWidth / 2 + 'px';}return () => {// 清理操作,如果需要的话};}, []); // 空依赖项数组,表示只在首次渲染和卸载时执行return <div ref={elementRef}>My Element</div>;
    }
    
  6. useState

    • 作用:用于在函数组件中添加状态。useState 返回一个状态变量和一个更新该状态的函数。
    • 与生命周期的关系:虽然 useState 本身不直接对应任何生命周期方法,但它使得函数组件能够拥有状态,从而可以响应状态变化并重新渲染。在某种程度上,可以认为状态更新触发了类似于 componentDidUpdate 的重新渲染过程。
  7. useMemouseCallback

    • 作用useMemo 用于缓存计算结果,避免在每次渲染时都重新计算。useCallback 用于缓存函数,避免在每次渲染时都重新创建函数实例。
    • 与生命周期的关系:这两个钩子并不直接对应生命周期方法,但它们有助于优化性能,减少不必要的计算和函数创建,从而间接影响组件的渲染性能。

需要注意的是,虽然 Hooks 提供了类似类组件生命周期的功能,但它们的使用方式和类组件的生命周期方法有所不同。Hooks 强调函数式编程的思想,通过纯函数和副作用分离来提高代码的可读性和可维护性。因此,在开发过程中,开发者需要根据具体场景选择合适的 Hooks 来实现所需的功能。


http://www.ppmy.cn/embedded/153088.html

相关文章

从零用java实现 小红书 springboot vue uniapp (8)个人资料修改 消息页优化

前言 移动端演示 http://8.146.211.120:8081/#/ 前面的文章我们主要完成了点赞关注 im 聊天功能 下面我们将完善个人资料修改 和消息页优化 向产品迈进 个人资料修改 自定义头像背景图 以及网名等等修改之后个人资料卡片也会随之改变 这样看起来就美观多了 修改文字资料的话很…

uni-app持久化登录简单实现

想要实现持久化登录&#xff0c;原理就是在每次进入应用的时候获取上一次用户登录的信息。 那么就好办了&#xff0c;我们在每次登录成功后把用户的账号密码存储到本地&#xff0c;然后在进入应用的时候读取本地文件获取账号密码重新执行登录流程&#xff0c;在退出登录的时候删…

【LeetCode】每日一题 2024_1_10 统计重新排列后包含另一个字符串的子字符串数目 II(滑动窗口)

前言 每天和你一起刷 LeetCode 每日一题~ 拼尽全力无法战胜期末考试 寒假 . . . 堂堂复活&#xff01; 每日一题重出江湖&#xff01; 就用我最擅长的滑动窗口类型的每日一题作为我寒假回归的第一题&#xff01; LeetCode 启动&#xff01; 题目&#xff1a;统计重新排列…

【ArcGIS微课1000例】0137:色彩映射表转为RGB全彩模式

本文讲述ArcGIS中,将tif格式的影像数据从色彩映射表转为RGB全彩模式。 参考阅读:【GlobalMapper精品教程】093:将tif影像色彩映射表(调色板)转为RGB全彩模式 文章目录 一、色彩映射表预览二、色彩映射表转为RGB全彩模式一、色彩映射表预览 加载配套数据包中的0137.rar中的…

Spring-Cloud-Gateway-Samples,nacos为注册中心,负载均衡

背景&#xff1a;本想找个简单例子看下&#xff0c;无奈版本依赖太过复杂&#xff0c;花了点时间。记录下吧 使用Spring Cloud Gateway作为网关服务&#xff0c;Nacos作为注册中心&#xff0c;实现对子服务的负载均衡访问。简单例子。 一、gateway-main-nacos服务端&#xff…

OpenSeaOtter架构设计

OpenSeaOtter的初衷是提供一个易于部署和使用的镜像存储服务&#xff0c;并且把研发流程中的CI/CD连接起来。 OpenSeaOtter可以接收多个格式的镜像&#xff0c;支持docker和podman 的pull和push。在镜像发生变更后&#xff0c;可以触发对应的部署行为。 架构 代码地址 我们的项…

《探秘鸿蒙NEXT中的人工智能核心架构》

在当今科技飞速发展的时代&#xff0c;华为HarmonyOS NEXT的发布无疑是操作系统领域的一颗重磅炸弹&#xff0c;其将人工智能与操作系统深度融合&#xff0c;开启了智能新时代。那么&#xff0c;鸿蒙NEXT中人工智能的核心架构究竟是怎样的呢&#xff1f;让我们一同探秘。 基础…

探索ScriptEcho:前端开发的神奇助手

《探索ScriptEcho&#xff1a;前端开发的神奇助手》 在前端开发的世界里&#xff0c;效率和准确性一直是开发者们追求的目标。今天&#xff0c;我要向大家介绍一款令人惊艳的工具——ScriptEcho&#xff0c;它可能会彻底改变前端开发的工作流程。 一、ScriptEcho的神奇特性 …