React 常用 Hooks 详细解析

ops/2025/3/14 8:45:19/

React Hooks 是 React 16.8 引入的核心特性,允许在函数组件中使用状态、生命周期等特性,替代类组件的复杂逻辑。以下是常用 Hooks 的详细解析及最佳实践:


1. useState:状态管理

用途:在函数组件中定义和更新局部状态。
示例

import { useState } from 'react';const Counter = () => {const [count, setCount] = useState(0); // 初始值 0return (<button onClick={() => setCount(count + 1)}>Clicked {count} times</button>);
};

注意

  • 状态更新是异步的,连续调用 setCount(count + 1) 不会立即生效。
  • 若新状态依赖旧状态,应使用函数形式:setCount(prev => prev + 1)

2. useEffect:副作用处理

用途:处理组件生命周期中的副作用(如数据请求、DOM 操作、订阅)。
示例

import { useEffect, useState } from 'react';const DataFetcher = ({ url }) => {const [data, setData] = useState(null);useEffect(() => {const fetchData = async () => {const response = await fetch(url);setData(await response.json());};fetchData();}, [url]); // 依赖项:url 变化时重新执行return <div>{data ? data.name : 'Loading...'}</div>;
};

注意

  • 依赖项数组
    • 空数组 []:仅在组件挂载时执行(类似 componentDidMount)。
    • 无数组:每次渲染后都执行(慎用)。
    • 包含变量:变量变化时重新执行。
  • 清理函数:返回一个函数用于清理(如取消订阅、移除事件监听):
    useEffect(() => {const timer = setInterval(() => {}, 1000);return () => clearInterval(timer); // 组件卸载时清理
    }, []);
    

3. useContext:跨组件数据传递

用途:在组件树中共享数据,避免逐层传递 props。
示例

import { createContext, useContext } from 'react';// 1. 创建 Context
const ThemeContext = createContext('light');// 2. 提供数据
const App = () => (<ThemeContext.Provider value="dark"><Toolbar /></ThemeContext.Provider>
);// 3. 消费数据
const Toolbar = () => {const theme = useContext(ThemeContext);return <div>Current theme: {theme}</div>;
};

4. useReducer:复杂状态逻辑

用途:类似 Redux 的状态管理,适合多状态关联或复杂更新逻辑。
示例

import { useReducer } from 'react';const initialState = { count: 0 };function reducer(state, action) {switch (action.type) {case 'increment':return { count: state.count + 1 };case 'decrement':return { count: state.count - 1 };default:throw new Error();}
}const Counter = () => {const [state, dispatch] = useReducer(reducer, initialState);return (<>Count: {state.count}<button onClick={() => dispatch({ type: 'increment' })}>+</button><button onClick={() => dispatch({ type: 'decrement' })}>-</button></>);
};

适用场景:状态更新涉及多个子值、依赖前一个状态、需要集中化管理。


5. useCallbackuseMemo:性能优化

useCallback:缓存函数
const handleClick = useCallback(() => {console.log('Clicked:', count);
}, [count]); // count 变化时重新创建函数
useMemo:缓存计算结果
const expensiveValue = useMemo(() => {return computeExpensiveValue(a, b);
}, [a, b]); // a/b 变化时重新计算

最佳实践

  • 仅当子组件依赖这些值且用 React.memo 优化时使用。
  • 避免滥用,内存缓存本身也有开销。

6. useRef:持久化引用

用途

  • 访问 DOM 元素。
  • 保存可变值(类似类组件的实例变量),不会触发重新渲染。
    示例
const TextInput = () => {const inputRef = useRef(null);const focusInput = () => {inputRef.current.focus();};return (<><input ref={inputRef} /><button onClick={focusInput}>Focus</button></>);
};

7. useLayoutEffect:同步副作用

用途:与 useEffect 类似,但会在 DOM 更新后同步执行(在浏览器绘制前)。
适用场景:需要直接操作 DOM 并确保用户看不到中间状态(如测量元素尺寸)。
示例

useLayoutEffect(() => {const { width } = divRef.current.getBoundingClientRect();setWidth(width); // 确保在渲染前获取最新尺寸
}, []);

8. 自定义 Hook:逻辑复用

用途:将组件逻辑封装为可复用的函数。
示例:自定义 useFetch

const useFetch = (url) => {const [data, setData] = useState(null);useEffect(() => {const fetchData = async () => {const response = await fetch(url);setData(await response.json());};fetchData();}, [url]);return data;
};// 使用
const MyComponent = () => {const data = useFetch('/api/data');return <div>{data}</div>;
};

Hooks 使用规则

  1. 只在顶层调用:不能在条件、循环或嵌套函数中使用 Hooks。
  2. 仅在 React 函数组件或自定义 Hook 中使用

总结

Hook核心用途典型场景
useState管理组件内部状态计数器、表单输入
useEffect处理副作用(数据请求、订阅)API 调用、事件监听
useContext跨组件共享数据主题、用户身份全局传递
useReducer复杂状态逻辑管理表单多字段、状态机
useCallback缓存函数,避免子组件无效渲染传递回调函数给优化过的子组件
useMemo缓存计算结果,减少重复计算复杂计算、优化渲染性能
useRef访问 DOM 或保存可变引用输入框聚焦、保存定时器 ID
useLayoutEffect同步 DOM 操作测量元素尺寸、强制同步更新

合理使用 Hooks 能显著提升代码可读性和可维护性,但需注意避免过度优化和滥用内存缓存。


http://www.ppmy.cn/ops/165622.html

相关文章

鸿蒙开发者社区资源的重要性

鸿蒙系统&#xff0c;作为华为公司自主研发的操作系统&#xff0c;旨在为各类智能设备提供统一的平台。它不仅支持手机、平板电脑等移动设备&#xff0c;还涵盖了物联网&#xff08;IoT&#xff09;设备和其他智能家居产品。鸿蒙系统的开发环境和工具链对于开发者来说至关重要&…

《又是二叉树?递归与回溯的经典应用》

“ 我喜欢晴天&#xff0c;你恰好是最好的太阳” 226.翻转二叉树 力扣题目链接(opens new window) 翻转一棵二叉树。 这道题我们可以通过递归法解决&#xff0c;我们只要递归的把每一个节点的左右孩子反转一下就能解决了。 代码如下&#xff1a; var invertTree function(ro…

如何通过数据分析提升美容院顾客复购率

在美容行业&#xff0c;顾客的复购率和流失率是衡量一家美容院经营状况的重要指标。然而&#xff0c;许多美容院在管理顾客关系时&#xff0c;常常面临复购率低、流失率高、数据统计困难等问题。根据《美容行业顾客行为分析》期刊的研究&#xff0c;超过60%的美容院在顾客关系管…

MySQL数据库操作

目录 SQL语句 1、SQL的背景 2、SQL的概念 SQL的分类 SQL的书写规范 MySQL数据库 1、MySQL数据库的编码 &#xff08;1&#xff09;utf8和utf8mb4的区别 &#xff08;2&#xff09;MySQL的字符集 &#xff08;3&#xff09;MySQL默认编码为 latin1 &#xff0c;如何更改…

本地部署 DeepSeek R1 的几种常见方法

本地部署 DeepSeek R1 的几种常见方法 使用 Ollama 部署使用量化模型部署使用 Docker 部署&#xff08;推荐&#xff09;硬件与性能要求其他工具 使用 Ollama 部署 安装 Ollama&#xff1a;访问 Ollama 官网&#xff0c;根据系统类型下载并安装。启动 Ollama&#xff1a;安装完…

stm32中分析UART中IDLE,RXNE,TC,TXE这些标志位的作用

下面将基于 STM32 标准库&#xff0c;结合之前提到的不同应用场景&#xff0c;给出使用 TXE、TC、IDLE 和 RXNE 标志位的代码示例及分析。 1. 连续数据发送&#xff08;使用 TXE&#xff09; 应用场景 向外部设备连续发送大量数据&#xff0c;如向显示屏发送显示数据、向传感…

Selenium Manager和webdriver manager的区别与联系

一、引言 1.1 自动化测试的重要性 在现代软件开发流程中&#xff0c;自动化测试已经成为保证软件质量和提高交付效率的关键实践。随着软件开发周期的缩短和软件复杂性的增加&#xff0c;手工测试已无法满足快速迭代的需求。自动化测试能够快速、准确地执行重复性测试任务&…

MS-DOS 6.22 下建立 FTP 服务器

在 MS-DOS 6.22 下建立 FTP 服务器需要使用第三方软件。以下是一个基本的步骤指南&#xff1a; 1. 下载并安装 mTCP 套件 mTCP 是一个适用于 MS-DOS 的网络工具包&#xff0c;其中包含一个 FTP 服务器程序。你可以从 mTCP 官方网站 下载该工具包。 2. 配置网络 确保你的 MS…