React自定义实现useWatch钩子的示例

 更新时间:2025年07月11日 10:58:53   作者:蓝精灵001  
React中实现类似Vue watch的useWatch Hook,包含基础监听、立即执行、动态停止及深度比较四种方式,适用于不同场景,文中通过示例代码介绍的非常详细,需要的朋友们下面随着小编来一起学习学习吧

如何实现类似Vue中watch功能的useWatch自定义Hook,以下是几种常见的实现方式和特点:

1. 基础实现

import { useEffect, useRef } from 'react';
 
function useWatch<T>(value: T, callback: (prev?: T) => void) {
  const prev = useRef<T>();
 
  useEffect(() => {
    callback(prev.current);
    prev.current = value;
  }, [value]);
}
  • 特点:简单直接,监听值变化并执行回调,同时提供旧值。
  • 使用场景:适用于大多数基本的监听需求。

2. 增加immediate选项

import { useEffect, useRef } from 'react';
 
type Config = {
  immediate: boolean;
};
 
function useWatch<T>(value: T, callback: (prev?: T) => void, config: Config = { immediate: false }) {
  const { immediate } = config;
  const prev = useRef<T>();
  const inited = useRef(false);
 
  useEffect(() => {
    const execute = () => callback(prev.current);
 
    if (!inited.current) {
      inited.current = true;
      if (immediate) {
        execute();
      }
    } else {
      execute();
    }
    prev.current = value;
  }, [value]);
}
  • 特点:增加了immediate选项,允许在组件初始化时立即执行回调。
  • 使用场景:当需要在组件加载时立即执行监听逻辑时非常有用。

3. 增加停止监听功能

import { useEffect, useRef } from 'react';
 
type Config = {
  immediate: boolean;
};
 
function useWatch<T>(value: T, callback: (prev?: T) => void, config: Config = { immediate: false }) {
  const { immediate } = config;
  const prev = useRef<T>();
  const inited = useRef(false);
  const stop = useRef(false);
 
  useEffect(() => {
    const execute = () => callback(prev.current);
 
    if (!stop.current) {
      if (!inited.current) {
        inited.current = true;
        if (immediate) {
          execute();
        }
      } else {
        execute();
      }
      prev.current = value;
    }
  }, [value]);
 
  return () => {
    stop.current = true;
  };
}
  • 特点:增加了停止监听的功能,允许动态控制监听的开启和关闭。
  • 使用场景:适用于需要在某些条件下动态停止监听的场景。

4. 使用lodash进行深度比较

import { useEffect, useRef } from 'react';
import { isEqual } from 'lodash';
 
function useWatch<T>(value: T, callback: (prev?: T) => void, config: Config = { immediate: false }) {
  const { immediate } = config;
  const prev = useRef<T>();
  const inited = useRef(false);
 
  useEffect(() => {
    const execute = () => {
      if (!isEqual(prev.current, value)) {
        callback(prev.current);
      }
    };
 
    if (!inited.current) {
      inited.current = true;
      if (immediate) {
        execute();
      }
    } else {
      execute();
    }
    prev.current = value;
  }, [value]);
}
  • 特点:使用lodashisEqual方法进行深度比较,确保监听的值是真正发生变化时才执行回调。
  • 使用场景:适用于监听复杂数据结构(如对象或数组)的变化。

使用示例

import React, { useState } from 'react';
import { useWatch } from './useWatch';
 
function App() {
  const [count, setCount] = useState(0);
 
  useWatch(count, (oldCount) => {
    console.log(`count changed from ${oldCount} to ${count}`);
  }, { immediate: true });
 
  return (
    <div>
      <p>Count: {count}</p>
      <button onClick={() => setCount(count + 1)}>Increment</button>
    </div>
  );
}
 
export default App;

总结

这些实现方式各有特点,可以根据具体需求选择合适的版本。如果需要简单的监听功能,基础实现就足够了;如果需要更复杂的控制,如立即执行或停止监听,则可以选择带有immediate和停止功能的版本。

到此这篇关于React自定义实现useWatch钩子的示例的文章就介绍到这了,更多相关React useWatch钩子内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

相关文章

  • 浅谈react.js 之 批量添加与删除功能

    浅谈react.js 之 批量添加与删除功能

    下面小编就为大家带来一篇浅谈react.js 之 批量添加与删除功能。小编觉得挺不错的,现在就分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2017-04-04
  • React实现PDF预览功能与终极优化

    React实现PDF预览功能与终极优化

    在前端开发中,PDF 预览是个常见需求,本文主要来带大家认识一个基于 react-pdf 的自定义 PDF 预览组件 PDFView,感兴趣的小伙伴可以了解下
    2025-05-05
  • 详解如何在React中执行条件渲染

    详解如何在React中执行条件渲染

    在现代Web开发中,React作为一种流行的JavaScript库,为开发者提供了一种高效构建用户界面的方式,条件渲染是React中的一个关键概念,本文将深入探讨如何在React中实现条件渲染,并通过示例代码演示常见的用法,需要的朋友可以参考下
    2025-01-01
  • React组件间通信的三种方法(简单易用)

    React组件间通信的三种方法(简单易用)

    React知识中一个主要内容便是组件之间的通信,以下列举几种常用的组件通信方式,本文就详细的介绍一下,感兴趣的可以了解一下
    2021-10-10
  • React组件中按钮的loading状态失效问题的解决方案

    React组件中按钮的loading状态失效问题的解决方案

    在React开发过程中,我遇到这样的情况:页面按钮的loading属性失效,尽管通过useEffect打印发现loading状态(确实在true和false之间切换,但按钮却没有表现出预期的加载效果,所以本文给大家介绍了失效的解决方案,需要的朋友可以参考下
    2025-07-07
  • 详解React中函数式组件与类组件的不同

    详解React中函数式组件与类组件的不同

    React 函数式组件与类组件的主要区别在于它们的定义和声明方式以及它们之间的一些特性,所以本文就详细的给大家讲讲React中函数式组件与类组件有何不同,需要的朋友可以参考下
    2023-09-09
  • react循环数据(列表)的实现

    react循环数据(列表)的实现

    这篇文章主要介绍了react循环数据的实现,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2021-04-04
  • reset.css浏览器默认样式表重置(user agent stylesheet)的示例代码

    reset.css浏览器默认样式表重置(user agent stylesheet)的示例代码

    这篇文章主要介绍了reset.css浏览器默认样式表重置(user agent stylesheet),本文通过示例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-12-12
  • React中Redux Hooks的使用详解

    React中Redux Hooks的使用详解

    这篇文章主要介绍了React Redux Hooks的使用详解,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友可以参考下
    2023-07-07
  • react实现antd线上主题动态切换功能

    react实现antd线上主题动态切换功能

    这篇文章主要介绍了react实现antd线上主题动态切换功能,本文给大家介绍的非常详细,具有一定的参考借鉴价值,需要的朋友可以参考下
    2019-08-08

最新评论