1

由于 useEffect 挂钩,我有一个组件可以在挂载时获取数据。我希望它不在挂载时重新获取数据,而是在我重新导航到该组件时使用 useSwr 挂钩提供的“缓存”数据。我不知道该怎么做。我读到的是你可以这样调用 swr :

const { data } = useSwr('same route as previous one')

它会为您提供前一个 swr 调用存储在缓存中的数据。


const CategoryList = ({setLoading}) => {

    const [category, setCategory] = useState('');
    const [mounted, setMounted] = useState(false);
    const [parameters, setParameters] = useState({});
    const company_id = localStorage.getItem('company_id')

    const session = new SessionService();
    const { dataFromFetch, error } = useSWR([mounted ? `${session.domain}/company/${company_id}/category-stats` : null, parameters ], url =>
      session.fetch(url, {
          method: 'GET',
      })
      , {
          onSuccess: (dataFromFetch) => {
            setCategory(dataFromFetch)
            setLoading(false)
            setMounted(false)
          },
          onError: (err, key, config) => {
            console.log("error", err)
          }
      }
  )

  useEffect(() => {
    setMounted(true)
    setLoading(true)
  }, [])

    return (
        <div className={classes.CategoryList}>
            <h5>Parc de véhicules</h5>
            <div className={classes.CategoriesCards}>
            {category.data? category.data.map((element, index) => {
                return <CategoryItem
                            category={element.data.name}
                            carNumber={element.stats.nb_vehicles}
                            locating={element.stats.nb_booked}
                            available={element.stats.nb_available}
                            blocked={element.stats.nb_unavailable}
                            percentage={(element.stats.nb_booked / element.stats.nb_vehicles * 100).toFixed(2)}
                            key={index}
                        />
            }): null}
            </div>
        </div>
    )
}

export default CategoryList;

另外,另一方面,我希望我的 SWR 挂钩不要一直尝试重新获取数据,就像它实际上正在做的那样。我尝试的是在我的 fetcher 函数之后传递选项,就像在这篇文章SWR 选项解释中规定的那样。实际上,我的组件正在尝试每 5-10 秒重新获取数据,但由于我的“已安装”条件导致“空”路由(这是根据文档推荐的方法)而失败。它仍然发送带有响应 404 的请求,我想避免这种情况。

const [parameters, setParameters] = useState({
      revalidateOnFocus: false,
      revalidateOnMount:false,
      revalidateOnReconnect: false,
      refreshWhenOffline: false,
      refreshWhenHidden: false,
      refreshInterval: 0
    });
    const company_id = localStorage.getItem('company_id')

    const session = new SessionService();
    const { dataFromFetch, error } = useSWR([mounted ? `${session.domain}/company/${company_id}/category-stats` : null, parameters ], url =>
      session.fetch(url, {
          method: 'GET',
      })
      , {
          onSuccess: (dataFromFetch) => {
            setCategory(dataFromFetch)
            setLoading(false)
            setMounted(false)
          },
          onError: (err, key, config) => {
            console.log("error", err)
          }
      }
  )
4

1 回答 1

2

根据 SWR 的文档,钩子的 API 是

const { data, error, isValidating, mutate } = useSWR(key, fetcher, options)

在您的代码中,您将选项作为第一个参数中的数组的一部分传递,而它应该是第三个。

一个小的重构显示了如何修复它:

const parameters = {
  revalidateOnFocus: false,
  revalidateOnMount: false,
  revalidateOnReconnect: false,
  refreshWhenOffline: false,
  refreshWhenHidden: false,
  refreshInterval: 0,
};

const company_id = localStorage.getItem( "company_id" );

const session = new SessionService();

const fetcher = (url) =>
  session.fetch(url, {
    method: "GET",
  });

const key = mounted
  ? `${session.domain}/company/${company_id}/category-stats`
  : null;

const { data, error } = useSWR(key, fetcher, parameters );

您还使您的代码过于复杂 - 不需要onSuccessonError处理程序 - 您可以简单地使用返回的值dataerror代替。

此外,无需使用setCategory. 直接从data. 这就是 SWR 的好处它会在获取数据时自动神奇地触发重新渲染。

于 2020-10-11T07:08:34.360 回答