如何使用 React Query 访问缓存以显示数据?

How to access cache to display data with React Query?

我的应用程序有一个用户输入一个 ID 作为请求发送,并且呈现并缓存与该 ID 匹配的响应数据。

我想访问缓存,这样我也可以显示它的数据,以便用户可以看到他们以前的查询。但是,我在尝试时收到此错误。 TypeError: Cannot read property 'getQueryData' of undefined.

以下是我访问缓存的尝试。

console.log(queryCache.getQueryData("rickandmorty"))
console.log(queryCache.getQueryData(["rickandmorty", idQuery]))
console.log(queryClient.getQueryData("rickandmorty"))
console.log(queryClient.getQueryData(["rickandmorty", idQuery]))

请让我知道我做错了什么。我正在使用版本 "react-query": "^3.13.0" https://codesandbox.io/s/rick-and-morty-2gy8r?file=/src/App.js

const [idQuery, setIdQuery] = useState(0);
const [rickAndMortyCharacter, setRickAndMortyCharacter] = useState({});
const queryRef = useRef(null);
const { register, handleSubmit, errors, setValue, watch } = useForm({
  resolver: yupResolver(searchSchema)
});
const formId = watch("rickAndMortyId");

const handleRickAndMortyFetch = () => {
  return delay()
    .then(() => axios(`https://rickandmortyapi.com/api/character/${idQuery}`))
    .then((res) => setRickAndMortyCharacter(res.data));
};

const { isLoading, error } = useQuery(
  ["rickandmorty", idQuery],
  handleRickAndMortyFetch,
  {
    refetchOnWindowFocus: false,
    enabled: idQuery !== 0,
    useErrorBoundary: true
  }
);
const disable = isLoading || parseFloat(formId) === idQuery;
const onSubmit = (formData) => {
  setIdQuery(formData.rickAndMortyId);
};

return (
  <div>
    <ErrorBoundary
      FallbackComponent={ErrorFallback}
      onReset={() => {
        queryRef.current.focus();
      }}
      resetKeys={[idQuery]}
    >
      <div>
        <form onSubmit={handleSubmit(onSubmit)}>
          <h3>Rick and Morty</h3>
          <input
            type="number"
            name="rickAndMortyId"
            placeholder="Between 1 and 671"
            ref={register}
            disabled={isLoading}
          />
          {errors.rickAndMortyId && <span>This field is required</span>}
          {error && <p>Error occurred: {error.message}</p>}
          <button type="submit" disabled={disable}>
            Search Character
          </button>
        </form>
        <button
          onClick={() => {
            const randomId = getRandomRickAndMortyCharacterId();
            setValue("rickAndMortyId", randomId);
            setIdQuery(randomId);
          }}
        >
          Random
        </button>
      </div>
      <div>
        {isLoading && <p>Loading...</p>}
        <RickAndMortyInfo rickAndMortyCharacter={rickAndMortyCharacter} />
      </div>
      <ReactQueryDevtools initialIsOpen={true} />
    </ErrorBoundary>
  </div>
);

在此代码中:

const [rickAndMortyCharacter, setRickAndMortyCharacter] = useState({});
const handleRickAndMortyFetch = () => {
  return delay()
    .then(() => axio('...'))
    .then((res) => setRickAndMortyCharacter(res.data));
};

const { isLoading, error } = useQuery(
  ["rickandmorty", idQuery],
  handleRickAndMortyFetch
);

成功获取后,将结果分配给本地状态而不是交给 react-query,因此它不知道应该缓存哪些数据。您需要 return 这样的结果:

const handleRickAndMortyFetch = () => {
  return delay()
    .then(() => axio('...'))
    .then((res) => res.data); // return instead of calling setState()
};

所以 react-query 可以 'catch' 并将其放入缓存中。下一步是删除本地状态 rickAndMortyCharacter 并改为引用 data。获取成功后的解析结果

const {
  isLoading,
  error,
  data // remove rickAndMortyCharacter and reference this instead
} = useQuery(
  ["rickandmorty", idQuery],
  handleRickAndMortyFetch
);

只有这样才能正常记录缓存:

queryClient.getQueryData(["rickandmorty", 518]);

现场演示