React Hooks深度历险记:从内置到自定义,揭秘魔法背后的原理

134 阅读16分钟

React Hooks深度历险记:从内置到自定义,揭秘魔法背后的原理

在React的世界里,Hooks就像一把开启魔法之门的钥匙,让函数组件拥有了状态管理、生命周期等超能力。今天,就让我们一起踏上这场深度历险,探索React Hooks的奥秘!

情侣专用.gif

一、缘起:为什么要有Hooks?

在Hooks出现之前,函数组件被戏称为"无状态组件",只能作为展示组件使用。复杂的状态逻辑和生命周期不得不交给类组件处理。但类组件也有自己的痛点:this绑定问题、难以复用的状态逻辑、复杂的生命周期方法等等。

于是,React团队在16.8版本引入了Hooks,它允许你在函数组件中使用状态和其他React特性。从此,函数组件一统江湖!

image.png

二、内置Hooks:React的魔法工具箱

1. useContext:跨越层级的通信使者

使用场景

当组件层级过深(父->子->孙->曾孙...),传统的props传递变得机械化且繁琐。useContext就是为解决这个痛点而生的。

useContext 是 React 中跨层级共享状态的一种方式,它让你在子组件中无需层层传递 props,就能访问祖先组件提供的数据。

工作原理
// 创建上下文
import { createContext } from 'react';
// 使用createContext创建一个名为ThemeContext的新上下文,并设置默认值为'light'
export const ThemeContext = createContext('light'); // 默认值

// 在顶层提供数据
function App() {
  const [theme, setTheme] = useState('light');
  return (
    <ThemeContext.Provider value={theme}>
      <Page />
      <button onClick={() => setTheme('dark')}>切换主题</button>
    </ThemeContext.Provider>
  );
}
  1. Context 创建: 使用 createContext(defaultValue) 创建一个上下文对象,它包含两个核心组件:

    • <MyContext.Provider>:用来提供值
    • <MyContext.Consumer>:用于消费值(类组件中使用较多)
  2. Provider 提供值: 在父组件中使用 <MyContext.Provider value={xxx}> 包裹子组件,把值传下去。

  3. useContext 消费值: 在任意子组件中调用 useContext(MyContext),即可获取最近的 Provider 提供的值。

  4. 自动更新: 当 Provider 的 value 变化时,所有使用该 Context 的子组件都会重新渲染并拿到最新值。


使用方法

步骤 1:创建 Context(通常单独文件导出)

步骤 2:在顶层组件中使用 Provider 提供值

步骤 3:在任意子组件中使用 useContext 获取值

注意事项
  • useContext 必须配合 createContext 和 Provider 使用。
  • 它只能在 React 函数组件或自定义 Hook 中使用。
  • 如果 Provider 的 value 是对象或函数,建议用 useMemo 优化避免不必要的重渲染。

进阶技巧:封装自定义Hook
//ThemeContext.js
//主题 适合全局
//创建一个上下文
import { createContext } from 'react';
//接受一个默认值,返回一个上下文对象Context
export const ThemeContext = createContext("light");

// hooks/useTheme.js
import { useContext } from "react";
import { ThemeContext } from "../ThemeContext";

export function useTheme() {
    return useContext(ThemeContext);
}
//App.jsx
import { useState } from 'react'
import { ThemeContext} from './ThemeContext'
function App() {
  const [theme, setTheme] = useState('light')
  console.log(ThemeContext,'//////')
  return (
    <ThemeContext.Provider value={theme}>
      <Page />
      <button onClick={() => setTheme("dark")}>切换主题</button>
    </ThemeContext.Provider>
  )
}

export default App

// 在组件中使用
const Page = () => {
    const theme = useTheme();
    return <div>{theme}</div>;
};
原理揭秘

useContext订阅最近的Context Provider的值。当Provider的值变化时,所有订阅该Context的组件都会重新渲染。

⚠️ 注意:如果Provider的值是对象,建议使用useMemo管理以避免不必要的渲染

2. useState:函数组件的状态引擎

基础用法
const [count, setCount] = useState(0);
同步还是异步?

在React事件处理函数中,setState是异步的! React会对多次setState调用进行批处理:

const handleClick = () => {
  setCount(count + 1);
  setCount(count + 1); // 注意:两次都基于同一个count值
  // 结果count只增加1,而不是2
};
函数式更新:解决异步陷阱
const handleClick = () => {
  setCount(prev => prev + 1);
  setCount(prev => prev + 1); // 基于最新状态值
  // 结果count增加2
};

使用函数式更新可以避免上述问题,因为它总是接收最新的状态值作为参数。具体来说,当使用函数式更新时,React 会将当前最新的状态值传递给你的更新函数。这里,每个 setCount 调用都提供了一个函数,该函数接受前一个状态值 prev 作为参数,并返回新的状态值。尽管 React 可能会批量处理这些更新,但因为每个更新都是基于传递进来的最新状态值来进行的,所以无论更新是如何被批处理的,最终的结果总是正确的:count 增加了 2

原理揭秘

React内部为每个组件维护一个"Hooks链表"。useState调用时会读取当前Hook节点的状态,并将指针移动到下一个节点。这就是为什么Hooks必须在顶层调用!

3. useEffect vs useLayoutEffect:副作用处理双雄

副作用

在 React 中,副作用(side effect)指的是那些不直接与渲染相关的逻辑操作。这些操作包括但不限于:

  • 数据获取
  • 手动 DOM 操作
  • 设置订阅
  • 日志记录
  • 定时器设置或清除
核心区别
特性useEffectuseLayoutEffect
执行时机浏览器绘制后异步执行DOM更新后同步执行
是否阻塞渲染
适用场景数据获取、订阅等DOM测量、防止闪烁等
useEffect使用示例
基本形式
useEffect(() => {
  // 副作用逻辑

  return () => {
    // 清理逻辑(可选)
  };
}, [/* 依赖项数组 */]);
  • 第一个参数:一个包含副作用逻辑的函数。
  • 返回值(可选) :一个清理函数,在组件卸载前或下次 Effect 运行之前执行。
  • 第二个参数(依赖项数组) :指定哪些状态变化时重新运行 Effect。
主要用途
  1. 数据获取

    useEffect(() => {
      fetch('https://api.example.com/data')
        .then(response => response.json())
        .then(data => setData(data));
    }, []); // 仅在挂载时执行
    
  2. 设置和清除订阅

    useEffect(() => {
      const subscription = props.source.subscribe();
      return () => subscription.unsubscribe(); // 清理订阅
    }, [props.source]); // 当 props.source 变化时重新订阅
    
  3. 定时器操作

    useEffect(() => {
      const timer = setInterval(() => setCount(prev => prev + 1), 1000);
      return () => clearInterval(timer); // 清除定时器
    }, []); // 仅在挂载和卸载时执行
    
关键点
  • 默认行为:每次渲染后执行。

  • 依赖项数组

    • 空数组:仅在挂载和卸载时执行。
    • 具体依赖项:当这些依赖项发生变化时执行。
  • 性能优化:通过依赖项数组控制 Effect 执行时机,避免不必要的重复执行。

总结
  • useEffect 是处理副作用的主要方式,适用于大多数场景。
  • 依赖项数组 控制 Effect 的触发条件,确保只在需要时执行。
  • 清理函数 避免内存泄漏和其他潜在问题。
useLayoutEffect解决闪烁问题

useLayoutEffect 是 React 中用于处理同步副作用的 Hook,它的用法与 useEffect 完全一致,但执行时机不同。

基本形式
useLayoutEffect(() => {
  // 同步副作用逻辑(如DOM操作)

  return () => {
    // 清理逻辑(可选)
  };
}, [/* 依赖项数组 */]);
⏱️ 执行时机
  • 在 DOM 更新后、浏览器绘制前立即同步执行。
  • 比 useEffect 更早执行(useEffect 是异步延迟执行)。
✅ 适用场景

当你需要:

  • 读取 DOM 布局(如宽高、位置)
  • 同步更新状态以避免视觉闪烁
  • 确保某些操作在页面重绘前完成

例如:

useLayoutEffect(() => {
  const element = document.getElementById('box');
  console.log(element.offsetWidth); // 获取真实宽度
}, []);

❗注意事项
  • 因为是同步执行,会阻塞页面渲染,影响性能。
  • 只有在确实需要同步操作时才使用它,否则优先使用 useEffect

如果只是普通副作用(如数据请求、事件监听),推荐使用 useEffect;若必须在页面渲染前进行 DOM 操作,就使用 useLayoutEffect

4. useReducer:状态管理的进阶方案

🛠️ useReducer 简介

useReducer 是 React 中用于管理复杂状态逻辑的一个 Hook,它通过分发动作(actions)来更新状态,类似于 Redux 的工作方式。

基本形式
const [state, dispatch] = useReducer(reducer, initialState);
  • reducer:一个函数,接收当前状态和动作作为参数,并返回新的状态。
  • initialState:状态的初始值。
  • state:当前状态。
  • dispatch:用于发送动作以更新状态。
⚙️ 工作原理
  1. 定义 Reducer 函数

    javascript
    深色版本
    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();
      }
    }
    
  2. 使用 useReducer

    javascript
    深色版本
    const [state, dispatch] = useReducer(reducer, { count: 0 });
    
  3. 派发动作

    javascript
    深色版本
    <button onClick={() => dispatch({ type: 'increment' })}>+</button>
    <button onClick={() => dispatch({ type: 'decrement' })}>-</button>
    
🎯 适用场景
  • 复杂的状态逻辑:当状态更新依赖于先前状态时。
  • 可预测的状态管理:适合需要明确状态变化流程的应用。

例如,管理计数器的状态:

function Counter() {
  const [state, dispatch] = useReducer(reducer, { count: 0 });

  return (
    <>
      Count: {state.count}
      <button onClick={() => dispatch({ type: 'increment' })}>+</button>
      <button onClick={() => dispatch({ type: 'decrement' })}>-</button>
    </>
  );
}
❗ 注意事项
  • 简单状态:对于简单的状态逻辑,直接使用 useState 更为简便。
  • 复杂状态:当状态逻辑变得复杂时,useReducer 提供了更清晰、更易于维护的状态管理方案。

useReducer 提供了一种结构化的方式来管理状态变化,特别适用于那些状态逻辑较为复杂的情况。对于简单的状态管理,建议继续使用 useState。如果需要处理复杂的状态流,useReducer 是一个强大的工具。

原理揭秘

useReduceruseState共享底层实现,都是基于React的Hooks链表机制。当调用dispatch时,React会计算新状态并触发重新渲染。

useRef

🔍 useRef 简介

useRef 是 React 中用于直接访问 DOM 元素或组件实例的 Hook,也可以用来保存任何可变值,其值在整个重新渲染过程中保持不变。

基本形式
const refContainer = useRef(initialValue);
  • refContainer.current:访问当前引用的值或 DOM 元素。
  • initialValue:初始值(通常是 null 或一个 DOM 节点)。

📌 主要用途
1. 访问 DOM 元素
const inputElement = useRef(null);

function focusInput() {
  inputElement.current.focus();
}

return (
  <>
    <input ref={inputElement} />
    <button onClick={focusInput}>Focus Input</button>
  </>
);
  • 用途

    • 这段代码展示了如何使用 useRef 来直接访问并操作 DOM 元素。在这里,inputElement 是一个引用对象,它通过 ref 属性绑定到了 <input> 元素上。
    • 当点击按钮触发 focusInput 函数时,inputElement.current 就指向了这个 <input> 元素,并调用 .focus() 方法使其获得焦点。
  • 执行 focusInput() 的作用

    • 调用 focusInput() 函数会使页面上的输入框自动聚焦,用户无需手动点击输入框即可开始输入。这对于提升用户体验(例如表单验证失败后自动将焦点移至第一个错误字段)非常有用。
2. 保存任意可变值
const count = useRef(0);

function incrementCount() {
  count.current += 1;
  console.log(count.current);
}
  • 用途

    • 使用 useRef 来保存一些在组件生命周期内需要持久化的数据,但这些数据的变化不应引起组件的重新渲染。与 useState 不同,修改 useRef 中的数据不会触发组件更新。
    • 在这里,count.current 用来存储一个计数器的值,每当调用 incrementCount 函数时,该值会增加 1 并打印到控制台。
  • count.current 的值是否增加了?

    • 是的,每次调用 incrementCount 函数时,count.current 的值确实会增加 1。但是,由于 useRef 的特性,这种改变不会导致组件重新渲染。这意味着如果你希望在 UI 上反映这一变化,你仍然需要使用 useState 或其他状态管理方式来同步视图。
  • 访问 DOM 元素useRef 提供了一种直接操作 DOM 的方式,适用于需要对特定元素进行精细控制的情况,如获取焦点、滚动位置调整等。
  • 保存任意可变值:虽然 useRef 可以保存和修改值,但它不会触发组件的重新渲染。这使得它非常适合用于那些不需要立即反映在界面中的临时或内部状态管理,比如计时器 ID、记录某些操作的发生次数等。
⏳ 生命周期
  • 挂载时useRef 初始化为指定的初始值。
  • 更新时useRef 的值在重新渲染之间保持不变,不会触发重新渲染。
❗ 注意事项
  • 避免滥用:尽量使用声明式的方式来处理 DOM 操作,只有在必要时才使用 useRef
  • 非响应式useRef 的值变化不会导致组件重新渲染,如果需要响应式的更新状态,请使用 useState
📌 总结
特性useRef
用途访问 DOM 元素或保存任意可变值
是否触发重新渲染
适用场景DOM 操作和保存非响应式状态

useRef 提供了一种方便的方式来直接访问 DOM 元素或保存一些跨渲染周期的值,而不会触发组件的重新渲染。对于需要手动操作 DOM 或保存临时状态而不引起视图更新的情况非常有用。如果你只需要响应式地管理状态,建议使用 useState

三、自定义Hooks:逻辑复用的终极武器

什么是自定义Hook?

自定义Hook是一个以use开头的函数,它可以调用其他Hooks,用于封装可复用的状态逻辑。与其他的函数不同之处在于自定义Hooks可以持有数据状态,我们甚至可以将所有的数据状态都封装到Hooks里面,让组件非常干净,专注于渲染,更便于维护代码

自定义 Hooks 和普通函数之间的主要区别在于前者专门用于封装和重用 React 组件逻辑,并且可以直接访问 React 的 Hooks 和特性。而普通函数则更适合于通用的任务处理和计算,不依赖于 React 的特定功能。

黄金法则

  1. 名称必须以use开头
  2. 可以调用其他Hooks
  3. 每次调用都有独立的状态

经典案例:useFetch

function useFetch(url) {
  const [data, setData] = useState(null);
  const [loading, setLoading] = useState(true);
  const [error, setError] = useState(null);

  useEffect(() => {
    const fetchData = async () => {
      try {
        const response = await fetch(url);
        const json = await response.json();
        setData(json);
      } catch (err) {
        setError(err);
      } finally {
        setLoading(false);
      }
    };

    fetchData();
  }, [url]); // url变化时重新获取

  return { data, loading, error };
}

// 使用示例
const { data, loading } = useFetch('/api/user');

原理揭秘

自定义Hooks本质是普通函数,它不引入新特性,只是组合现有Hooks。每次调用都会在组件内部创建新的Hooks链表节点。

四、Fragment:隐形的包裹者

为什么需要Fragment?

在React中,组件必须返回单个根元素。Fragment允许你分组子元素而不增加额外DOM节点。

使用方式

// 显式语法
import { Fragment } from 'react';

function List() {
  return (
    <Fragment>
      <li>项目1</li>
      <li>项目2</li>
    </Fragment>
  );
}

// 简写语法(更常用)
function List() {
  return (
    <>
      <li>项目1</li>
      <li>项目2</li>
    </>
  );
}

关键特性

  • 不产生额外DOM节点
  • 支持key属性(循环中必需)
  • 简洁的语法糖

五、Hooks底层原理探秘

Hooks的链表结构

React内部使用链表来管理Hooks。这就是为什么:

  1. Hooks必须在顶层调用(不能嵌套在条件/循环中)
  2. Hooks调用顺序必须保持稳定
graph LR
A[组件首次渲染] --> B[创建Hooks链表]
C[后续渲染] --> D[按顺序遍历链表]

Hooks执行流程

  1. 初始化:组件首次渲染,创建Hooks链表
  2. 更新:后续渲染,按顺序读取链表中的Hook
  3. 清理:组件卸载时执行所有effect的清理函数

为什么Hooks依赖数组如此重要?

依赖数组决定了effect何时重新执行。React使用浅比较来判断依赖项是否变化:

useEffect(() => {
  // 仅当id变化时执行
}, [id]); 

 为什么 Hooks 必须在顶层调用?

Hooks 必须在 React 函数组件的顶层调用,不能在条件语句、循环或其他嵌套函数中调用。这是因为:

  • 保持 Hook 调用顺序一致:每次组件渲染时,React 需要按照相同的顺序调用所有的 Hooks。这样它才能正确地匹配并恢复之前的状态。如果 Hook 调用顺序发生变化,比如在一个条件分支中跳过了一些 Hook,那么 React 就无法正确地找到对应的 Hook 状态,导致状态混乱。
  • 确保 Hook 索引正确:由于 React 使用索引来追踪每个 Hook 的位置,如果 Hook 的调用顺序变化了,索引就会错位,导致错误的状态被读取或更新。

 具体示例说明

考虑以下不正确的使用方式:

function Counter({ condition }) {
  const [count, setCount] = useState(0);

  if (condition) {
    const [anotherState, setAnotherState] = useState(1); // 错误!不应该在这里调用 useState
  }

  return (
    <div>
      Count: {count}
      <button onClick={() => setCount(count + 1)}>Increment</button>
    </div>
  );
}

在这个例子中,useState 被放在了一个条件语句内。这意味着当 conditionfalse 时,anotherState 这个 Hook 不会被调用。然而,当下一次 condition 变为 true 时,React 会尝试为 anotherState 分配状态,但由于之前的渲染过程中没有这个 Hook 的调用记录,React 无法正确处理这种情况,可能会导致状态丢失或混淆。

正确的做法应该是:


function Counter({ condition }) {
  const [count, setCount] = useState(0);
  const [anotherState, setAnotherState] = useState(1); // 正确!在顶层调用

  return (
    <div>
      Count: {count}
      <button onClick={() => setCount(count + 1)}>Increment</button>
      {condition && (
        <div>Another State: {anotherState}</div>
      )}
    </div>
  );
}

这里,所有 Hook 都是在组件的顶层调用的,无论 condition 是否为 true,Hook 的调用顺序始终保持不变。

总结

  • 一致性:保证每次渲染时 Hook 的调用顺序一致,以便 React 能够准确地恢复和更新状态。
  • 索引依赖性:React 依赖于 Hook 调用的索引来匹配状态,任何改变 Hook 调用顺序的操作都会破坏这种依赖关系,导致不可预测的行为。

遵循这一规则,可以帮助你避免许多潜在的问题,并确保你的 React 组件能够稳定且高效地运行。

六、最佳实践与性能优化

Hooks使用守则

  1. 只在React函数组件或自定义Hook中使用Hooks
  2. 保持Hooks调用顺序稳定
  3. 使用ESLint插件(eslint-plugin-react-hooks)避免常见错误

性能优化技巧

  1. 使用useCallback记忆函数,避免不必要的重新创建
  2. 使用useMemo记忆计算结果
  3. 合理拆分组件,隔离变化
// 优化示例
const memoizedCallback = useCallback(() => {
  doSomething(a, b);
}, [a, b]);

const memoizedValue = useMemo(() => computeExpensiveValue(a, b), [a, b]);

七、总结:Hooks的力量

通过本文的历险,我们深入了解了React Hooks的精髓:

Hook类别代表核心作用
状态管理useState函数组件的状态基石
useReducer复杂状态逻辑管理
上下文通信useContext跨组件层级通信
副作用处理useEffect异步副作用处理
useLayoutEffect同步DOM操作,防止闪烁
性能优化useCallback记忆函数引用
useMemo记忆计算结果
DOM操作useRef访问DOM/保存可变值
自定义HookuseXXX逻辑复用封装

Hooks的设计哲学是关注点分离逻辑复用。它让我们的代码更简洁、更易于维护。

黄金法则:只在最顶层使用Hooks,不要在循环、条件或嵌套函数中调用Hooks!

Hooks已经彻底改变了React开发方式。掌握它们,你将拥有构建现代化React应用的超能力!

Suggestion.gif