首页 > 解决方案 > 使用 useCallback 记住一个闭包和渲染回调

问题描述

让我有一个 graphql 突变组件,我在很多地方重复使用

const MarkAsViewed =({ type = 1, children }) => {
  const markAsViewed = (commitMutation) => (type) => {
    return commitMutation({
      variables: { type }
    });
  };

  return (
    <MarkAsViewedMutation
      mutation={MARK_AS_VIEWED_MUTATION}
      variables={{
        type,
      }}
    >
      {
        (commitMutation, { error, loading }) => children({
          markAsViewed: markAsViewed(commitMutation)
        })
      }
    </MarkAsViewedMutation>
  );
};

然而,由于markAsViewed是一个闭包函数,它总是会返回具有不同 ref 的不同函数,这意味着反应不同。

这使得子组件必须这样做useCallback

const alwaysSameRefFunc = useCallback(()=>{ markAsViewed(), []}

以上工作,但产生2个问题:

  1. 我收到 linter 警告说我应该添加markAsViewed为依赖等等。我不能,因为它会触发无限循环(因为每次都是不同的参考)
  2. 每个使用<MarkAsViewed />组件的人都需要手动记忆

理想情况下,这是我想要的,但它是无效代码,因为“markAsViewed”不是反应组件,不能有 useCallback

const markAsViewed = (commitMutation) => useCallback((type) => {
    return commitMutation({
      variables: { type }
    });
  }, []);

知道如何解决这个问题吗?

注意:我们还没有准备好将 Apollo 版本更新为 Hook

标签: reactjsgraphqlapollo

解决方案


下面的工作吗?

const markAsViewed = commitMutation => type => {
  return commitMutation({
    variables: { type },
  });
};
const MarkAsViewed = ({ type = 1, children }) => {
  const fn = useCallback(
    (commitMutation, { error, loading }) =>
      children({
        markAsViewed: markAsViewed(commitMutation),
      }),
    [children]
  );
  return (
    <MarkAsViewedMutation
      mutation={MARK_AS_VIEWED_MUTATION}
      variables={{
        type,
      }}
    >
      {fn}
    </MarkAsViewedMutation>
  );
};

我不确定这是否会起作用,因为它仍然取决于子级,如果这会导致不必要的渲染,那么可能会发布 MarkAsViewed 组件的渲染方式。


推荐阅读