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和停止功能的版本。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值