React自定义实现useWatch的方式和特点

 更新时间:2025年05月16日 11:35:57   投稿:mrr  
这篇文章主要介绍了React自定义实现useWatch的方式和特点,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧

如何实现类似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 Markdown配置示例

    React Markdown配置示例

    React-Markdown 是一个基于 React 的 Markdown 渲染组件库,其核心设计理念是通过 Unified 生态系统实现安全、可扩展的 Markdown 解析
    2025-04-04
  • react-native封装插件swiper的使用方法

    react-native封装插件swiper的使用方法

    这篇文章主要介绍了react-native封装插件swiper的使用方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-03-03
  • React教程之封装一个Portal可复用组件的方法

    React教程之封装一个Portal可复用组件的方法

    react的核心之一是组件,下面这篇文章主要给大家介绍了关于React教程之封装一个Portal可复用组件的相关资料,文中通过示例代码介绍的非常详细,需要的朋友可以参考借鉴,下面随着小编来一起学习学习吧。
    2018-01-01
  • 详解React 如何防止 XSS 攻击论$$typeof 的作用

    详解React 如何防止 XSS 攻击论$$typeof 的作用

    这篇文章主要介绍了详解React 如何防止 XSS 攻击论$$typeof 的作用,文章围绕主题展开详细的内容介绍,具有一定的参考价值,需要的小伙伴可以参考一下
    2022-07-07
  • 详解React之key的使用和实践

    详解React之key的使用和实践

    这篇文章主要介绍了详解React之key的使用和实践,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-09-09
  • React props的使用小结

    React props的使用小结

    本文主要介绍了React props的使用小结,文中通过示例代码介绍的非常详细,对大家的学习或者工作具有一定的参考学习价值,需要的朋友们下面随着小编来一起学习学习吧
    2025-09-09
  • React表单容器的通用解决方案

    React表单容器的通用解决方案

    本文主要介绍了React表单容器的通用解决方案,文中通过示例代码介绍的非常详细,具有一定的参考价值,感兴趣的小伙伴们可以参考一下
    2022-04-04
  • 使用 React hooks 实现类所有生命周期

    使用 React hooks 实现类所有生命周期

    react 自 16.8 开始,引入了 Hooks 概念,使得函数组件中也可以拥有自己的状态,并且可以模拟对应的生命周期,这篇文章主要介绍了使用 React hooks 怎么实现类里面的所有生命周期,需要的朋友可以参考下
    2023-02-02
  • create-react-app构建项目慢的解决方法

    create-react-app构建项目慢的解决方法

    这篇文章主要介绍了create-react-app构建项目慢的解决方法,小编觉得挺不错的,现在分享给大家,也给大家做个参考。一起跟随小编过来看看吧
    2018-03-03
  • Vue 3开发者转型React保姆级教程(含实际项目案例)

    Vue 3开发者转型React保姆级教程(含实际项目案例)

    从Vue转到React并不难,两者作为主流前端框架,核心思想(组件化、声明式编程)一致,但在语法、API设计和最佳实践上存在差,这篇文章主要介绍了Vue3开发者转型React保姆级教程的相关资料,需要的朋友可以参考下
    2026-05-05

最新评论