NextJs 应用程序在 SWR 中使用 JavaScript toISOString() 导致无限 API 调用
NextJs app using JavaScript toISOString() in SWR causing infinite API calls
我设置了以下示例 Next.js 应用程序来测试我在 useSWR
调用中包含的变量上使用 toISOString()
JavaScript 函数时遇到的问题到 API 终点:
下面是 pages/user/[id].tsx
的代码,其中 toISOString()
被调用:
import NextLink from "next/link";
import {
Link,
Flex,
Box,
Text,
SimpleGrid,
Heading,
Alert
} from "@chakra-ui/core";
import useSWR from "swr";
import { useRouter } from "next/router";
type Data = {
id: string;
name: string;
email: string;
};
const fetcher = async (url: string) => {
const res = await fetch(url);
if (!res.ok) {
throw Error("Yo that's NOT OK!!!");
}
const data: Data = await res.json();
return data;
};
const UserData = () => {
const router = useRouter();
const { id } = router.query;
const today = new Date();
const yesterday = new Date(today);
yesterday.setDate(yesterday.getDate() - 1);
const ssd = yesterday.toISOString(); // TROUBLE
// const ssd = "2021-08-16T19:27:51.630Z"; //fine
const result = useSWR(`/api/user/${id}/${ssd}`, fetcher, {
revalidateOnFocus: false
});
const data: Data = result.data;
const error: Error = result.error;
if (error) {
return <Alert status="error">Loading failed: {error.message}</Alert>;
}
if (!data) {
return <Alert status="info">Loading...</Alert>;
}
return (
<SimpleGrid columns={2} width="2xs" spacingY={4}>
<Text fontWeight="bold" marginRight={4}>
UserID
</Text>
<Text>{data.id}</Text>
<Text fontWeight="bold" marginRight={4}>
Name
</Text>
<Text>{data.name}</Text>
<Text fontWeight="bold" marginRight={4}>
Email
</Text>
<Text>{data.email}</Text>
</SimpleGrid>
);
};
const UserPage = () => {
return (
<Box>
<Flex flexDirection="column" alignItems="center">
<Heading marginY="2rem">User</Heading>
<UserData />
<NextLink href="/">
<Link marginY="2rem">
<Text fontStyle="italic">Go back home</Text>
</Link>
</NextLink>
</Flex>
</Box>
);
};
export default UserPage;
pages/user/[id].tsx
的第 34 行是调用 toISOString()
并触发多个 API 调用的地方,如浏览器控制台所示(控制台语句添加到 /pages/api/user/[...params].ts
第 7 行 -请参阅下面的完整示例代码)。 toISOString()
函数似乎在 Next.js 中充当 useState
函数,因此触发了似乎不会停止的 API 调用。
关于如何使用 toISOString()
并且不让它触发 useSWR()
的多个 API 调用的任何想法?
下面是完整示例应用程序的副本:
这是您更新后的代码,它解决了无限 API 调用的问题,只需将 ISOString 移至父级即可。
import NextLink from "next/link";
import {
Link,
Flex,
Box,
Text,
SimpleGrid,
Heading,
Alert
} from "@chakra-ui/core";
import useSWR from "swr";
import { useRouter } from "next/router";
type Data = {
id: string;
name: string;
email: string;
};
const fetcher = async (url: string) => {
const res = await fetch(url);
if (!res.ok) {
throw Error("Yo that's NOT OK!!!");
}
const data: Data = await res.json();
return data;
};
const UserData = ({ ssd }) => {
const router = useRouter();
const { id } = router.query;
// const ssd = "2021-08-16T19:27:51.630Z"; //fine
const result = useSWR(`/api/user/${id}/${ssd}`, fetcher, {
revalidateOnFocus: false
});
const data: Data = result.data;
const error: Error = result.error;
if (error) {
return <Alert status="error">Loading failed: {error.message}</Alert>;
}
if (!data) {
return <Alert status="info">Loading...</Alert>;
}
return (
<SimpleGrid columns={2} width="2xs" spacingY={4}>
<Text fontWeight="bold" marginRight={4}>
UserID
</Text>
<Text>{data.id}</Text>
<Text fontWeight="bold" marginRight={4}>
Name
</Text>
<Text>{data.name}</Text>
<Text fontWeight="bold" marginRight={4}>
Email
</Text>
<Text>{data.email}</Text>
</SimpleGrid>
);
};
const UserPage = () => {
const today = new Date();
const yesterday = new Date(today);
yesterday.setDate(yesterday.getDate() - 1);
const ssd = yesterday.toISOString(); // No More TROUBLE
return (
<Box>
<Flex flexDirection="column" alignItems="center">
<Heading marginY="2rem">User</Heading>
<UserData ssd={ssd} />
<NextLink href="/">
<Link marginY="2rem">
<Text fontStyle="italic">Go back home</Text>
</Link>
</NextLink>
</Flex>
</Box>
);
};
export default UserPage;
我设置了以下示例 Next.js 应用程序来测试我在 useSWR
调用中包含的变量上使用 toISOString()
JavaScript 函数时遇到的问题到 API 终点:
下面是 pages/user/[id].tsx
的代码,其中 toISOString()
被调用:
import NextLink from "next/link";
import {
Link,
Flex,
Box,
Text,
SimpleGrid,
Heading,
Alert
} from "@chakra-ui/core";
import useSWR from "swr";
import { useRouter } from "next/router";
type Data = {
id: string;
name: string;
email: string;
};
const fetcher = async (url: string) => {
const res = await fetch(url);
if (!res.ok) {
throw Error("Yo that's NOT OK!!!");
}
const data: Data = await res.json();
return data;
};
const UserData = () => {
const router = useRouter();
const { id } = router.query;
const today = new Date();
const yesterday = new Date(today);
yesterday.setDate(yesterday.getDate() - 1);
const ssd = yesterday.toISOString(); // TROUBLE
// const ssd = "2021-08-16T19:27:51.630Z"; //fine
const result = useSWR(`/api/user/${id}/${ssd}`, fetcher, {
revalidateOnFocus: false
});
const data: Data = result.data;
const error: Error = result.error;
if (error) {
return <Alert status="error">Loading failed: {error.message}</Alert>;
}
if (!data) {
return <Alert status="info">Loading...</Alert>;
}
return (
<SimpleGrid columns={2} width="2xs" spacingY={4}>
<Text fontWeight="bold" marginRight={4}>
UserID
</Text>
<Text>{data.id}</Text>
<Text fontWeight="bold" marginRight={4}>
Name
</Text>
<Text>{data.name}</Text>
<Text fontWeight="bold" marginRight={4}>
Email
</Text>
<Text>{data.email}</Text>
</SimpleGrid>
);
};
const UserPage = () => {
return (
<Box>
<Flex flexDirection="column" alignItems="center">
<Heading marginY="2rem">User</Heading>
<UserData />
<NextLink href="/">
<Link marginY="2rem">
<Text fontStyle="italic">Go back home</Text>
</Link>
</NextLink>
</Flex>
</Box>
);
};
export default UserPage;
pages/user/[id].tsx
的第 34 行是调用 toISOString()
并触发多个 API 调用的地方,如浏览器控制台所示(控制台语句添加到 /pages/api/user/[...params].ts
第 7 行 -请参阅下面的完整示例代码)。 toISOString()
函数似乎在 Next.js 中充当 useState
函数,因此触发了似乎不会停止的 API 调用。
关于如何使用 toISOString()
并且不让它触发 useSWR()
的多个 API 调用的任何想法?
下面是完整示例应用程序的副本:
这是您更新后的代码,它解决了无限 API 调用的问题,只需将 ISOString 移至父级即可。
import NextLink from "next/link";
import {
Link,
Flex,
Box,
Text,
SimpleGrid,
Heading,
Alert
} from "@chakra-ui/core";
import useSWR from "swr";
import { useRouter } from "next/router";
type Data = {
id: string;
name: string;
email: string;
};
const fetcher = async (url: string) => {
const res = await fetch(url);
if (!res.ok) {
throw Error("Yo that's NOT OK!!!");
}
const data: Data = await res.json();
return data;
};
const UserData = ({ ssd }) => {
const router = useRouter();
const { id } = router.query;
// const ssd = "2021-08-16T19:27:51.630Z"; //fine
const result = useSWR(`/api/user/${id}/${ssd}`, fetcher, {
revalidateOnFocus: false
});
const data: Data = result.data;
const error: Error = result.error;
if (error) {
return <Alert status="error">Loading failed: {error.message}</Alert>;
}
if (!data) {
return <Alert status="info">Loading...</Alert>;
}
return (
<SimpleGrid columns={2} width="2xs" spacingY={4}>
<Text fontWeight="bold" marginRight={4}>
UserID
</Text>
<Text>{data.id}</Text>
<Text fontWeight="bold" marginRight={4}>
Name
</Text>
<Text>{data.name}</Text>
<Text fontWeight="bold" marginRight={4}>
Email
</Text>
<Text>{data.email}</Text>
</SimpleGrid>
);
};
const UserPage = () => {
const today = new Date();
const yesterday = new Date(today);
yesterday.setDate(yesterday.getDate() - 1);
const ssd = yesterday.toISOString(); // No More TROUBLE
return (
<Box>
<Flex flexDirection="column" alignItems="center">
<Heading marginY="2rem">User</Heading>
<UserData ssd={ssd} />
<NextLink href="/">
<Link marginY="2rem">
<Text fontStyle="italic">Go back home</Text>
</Link>
</NextLink>
</Flex>
</Box>
);
};
export default UserPage;