在safari浏览器中,可能会遇到异步时间差的问题,导致状态没有及时更新到state,引起传参错误。
PS:由于useState是一个普通的函数, 定义为() => void;因此此处不能用await/async替代setTimeout,只能用在返回值为Promise时
有问题的代码:
我们可以看到,setTimeout只传0,这个时候在谷歌浏览器是有效果的,能确保顺序正确,但是在safari浏览器下,obj的值没有办法及时更新上state。
typescript
const columnsChange = ({ key, val }: { key: string | number; val: any }) => {
const obj = {
[key]: Array.isArray(val) && val?.length > 0 ? val : undefined
};
setFilters((f) => ({ ...f, ...obj }));
setPagination({ pageNo: 1, pageSize: PAGE_SIZE });
setTimeout(() => {
run();
}, 0);
};
修改后的代码:
于是,我们尝试使用await/async去处理异步,但是由于setState是一个普通的函数, 定义为() => void;因此此处不能用await/async替代setTimeout,只能用在返回值为Promise时。因此不能这样做!
typescript
const columnsChange = async ({ key, val }: { key: string | number; val: any }) => {
const obj = {
[key]: Array.isArray(val) && val?.length > 0 ? val : undefined
};
// 由于setState是一个普通的函数, 定义为() => void;因此此处不能用await/async替代setTimeout,只能用在返回值为Promise时
await setFilters((f) => ({ ...f, ...obj }));
setPagination({ pageNo: 1, pageSize: PAGE_SIZE });
run();
};
验证:我们通过查看useState的定义,查看Dispatch、SetStateAction的定义,可以看到都是() => void或者返回泛型S(这意味着如果我们要变更的状态不是promise,也不适用await写法)
正确的代码:
将异步时间微调大一些200ms
typescript
const columnsChange = ({ key, val }: { key: string | number; val: any }) => {
const obj = {
[key]: Array.isArray(val) && val?.length > 0 ? val : undefined
};
// 由于setState是一个普通的函数, 定义为() => void;因此此处不能用await/async替代setTimeout,只能用在返回值为Promise时
setFilters((f) => ({ ...f, ...obj }));
setPagination({ pageNo: 1, pageSize: PAGE_SIZE });
setTimeout(() => {
run();
}, 200);
};