检查 useEffect 中的哪个依赖项已更新

dan*_*son 6 reactjs use-effect

我有这个useEffect钩子:

useEffect(() => {
  setTop(t => t + (controller.position.y * tileSize))
  setLeft(l => l + (controller.position.x * tileSize))
}, [controller.position])
Run Code Online (Sandbox Code Playgroud)

我希望它只在发生position变化时才进行添加。如果发生tileSize变化,我只想让它进行乘法。

我尝试将其分成两部分useEffect,但随后收到React Hook useEffect has missing dependencies警告:

useEffect(() => {
    setTop(t => t + (controller.position.y * spriteSize))
    setLeft(l => l + (controller.position.x * spriteSize))
}, [controller.position])

useEffect(() => {
    setTop((controller.position.y * spriteSize))
    setLeft((controller.position.x * spriteSize))
}, [spriteSize])
Run Code Online (Sandbox Code Playgroud)

在这种情况下,最佳做法是什么?

编辑:

一个可重现的例子:

const [tileSize, setTileSize] = useState(0)
const controller = {
    position: {
        x: 0,
        y: 0
    }
}
useEffect(() => {
    setTop(t => t + (controller.position.y * tileSize))
    setLeft(l => l + (controller.position.x * tileSize))
}, [controller.position])

useEffect(() => {
    setTop((controller.position.y * tileSize))
    setLeft((controller.position.x * tileSize))
}, [tileSize])

const asdf = () => {
    setTileSize(150)
}

return (
    <div onClick={() => asdf()}>click me</div>
)
Run Code Online (Sandbox Code Playgroud)

警告信息:

第 31 行:React Hook useEffect 缺少依赖项:“tileSize”。包含它或删除依赖项数组。如果“setTop”需要“tileSize”的当前值,您还可以用 useReducer 替换多个 useState 变量

第 36 行:React Hook useEffect 缺少依赖项:“controller.position.x”和“controller.position.y”。要么包含它们,要么删除依赖数组react-hooks/exhaustive-deps 第 46 行:无法访问的代码 no-unreachable

Den*_*ash 1

方法有很多种,您应该针对您的用例寻找最具可读性的一种。

  1. 正如警告所示,替换为useReducer

React Hook useEffect 缺少依赖项:“tileSize”。包含它或删除依赖项数组。如果“setTop”需要“tileSize”的当前值,您还可以用 useReducer 替换多个 useState 变量。(反应钩子/详尽的依赖)

const STATE = {
  ADDITION: 'addition',
  MULTIPLICATION: 'multiplication'
};

function reducer(state, action) {
  switch (action.type) {
    case STATE.ADDITION:
      return {
        ...state,
        position: action.position,
        top: state.top + action.position * state.spriteSize
      };
    case STATE.MULTIPLICATION:
      return {
        ...state,
        spriteSize: action.spriteSize,
        top: state.position * action.spriteSize
      };
    default:
      throw new Error();
  }
}

function Controller({ position, spriteSize }) {
  const [state, dispatch] = useReducer(reducer, {
    top: 0,
    position,
    spriteSize
  });

  useEffect(() => {
    dispatch({ type: STATE.ADDITION, position });
  }, [position]);

  useEffect(() => {
    dispatch({ type: STATE.MULTIPLICATION, spriteSize });
  }, [spriteSize]);

  return <FlexBox>{state.top}</FlexBox>;
}
Run Code Online (Sandbox Code Playgroud)
  1. 您可以使用参考useRef并与旧值进行比较
  2. 如果您知道自己在做什么,只需禁用 lint 警告即可。

编辑 styled-antd-react-starter