为什么 useEffect 中的状态没有更新?
State is not updating in useEffect why?
我正在尝试制作一个加密货币价格跟踪器网站。我在每 10 秒更新一次 coinList 变量 prevData 变量后获取数据。我也想访问 useEffect 中的 prevData 变量,但在它里面变量没有更新,它的值仍然是 []。但是,如果我在 useEffect 之外使用 prevData,数据就会发生变化,我可以使用它。
我只想要正在更新的 useEffect 中 prevData 的值。
function App() {
const [coinsList, setCoinsList] = useState([]);
const [prevData, setPrevData] = useState([]);
let [width, setWidth] = useState(window.innerWidth)
useEffect(() => {
async function fetchData() {
const response = await fetch('api key').then(res => res.json())
setCoinsList(res);
setPrevData(res);
}
fetchData()
async function fetchData2() {
const res = await fetch('api key').then(res => res.json())
for (let i = 0; i < prevData.length; i++) {
if (res[i].current_price > prevData[i].current_price) {
let coin = document.getElementById(`${prevData[i].id}`)
coin.classList.add('green')
console.log(coin)
// setTimeout(() => {
// coin.classList.remove('green')
// }, 1000)
} else if (res[i].current_price < prevData[i].current_price) {
let coin = document.getElementById(`${prevData[i].id}`)
coin.classList.add('red')
console.log(coin)
// setTimeout(() => {
// coin.classList.remove('red')
// }, 1000)
} else {
let coin = document.getElementById(`${prevData[i].id}`)
console.log(coin)
coin.classList.add('blue')
// setTimeout(() => {
// coin.classList.remove('blue')
// }, 1000)
}
}
setPrevData(res);
setCoinsList(res);
}
setInterval(() => {
fetchData2()
console.log(prevData) // it is not updating
}, 10000)
}, [])
useEffect(() => {
window.addEventListener('resize', () => {
setWidth(window.innerWidth)
})
}, [window.innerWidth])
console.log(prevData) // it is updating
您在 setInterval
范围内的 prevData
值上有 closure,您可以使用 ref 修复它:
const prevDataRef = useRef();
// Keep the ref updates
useEffect(() => {
prevDataRef.current = prevData;
}, []);
// no closure
useEffect(() => {
setInterval(() => {
fetchData2(); // use prevDataRef.current
console.log(prevDataRef.current); // updated
}, 10000);
}, []);
在 fetchData2
中,您有一个 prevData
状态的陈旧外壳。它在间隔回调范围内关闭。
一种补救方法是使用 React ref 和额外的 useEffect
挂钩来缓存 prevData
状态的副本,可以在 fetchData2
回调中访问。
function App() {
const [coinsList, setCoinsList] = useState([]);
const [prevData, setPrevData] = useState([]);
const prevDataRef = React.useRef(prevData);
useEffect(() => {
prevDataRef.current = prevData;
}, [prevData]);
...
useEffect(() => {
...
async function fetchData2() {
const prevData = prevDataRef.current; // <-- grab current value
const res = await fetch('api key').then(res => res.json());
for (let i = 0; i < prevData.length; i++) {
if (res[i].current_price > prevData[i].current_price) {
let coin = document.getElementById(`${prevData[i].id}`);
coin.classList.add('green');
} else if (res[i].current_price < prevData[i].current_price) {
let coin = document.getElementById(`${prevData[i].id}`);
coin.classList.add('red');
} else {
let coin = document.getElementById(`${prevData[i].id}`);
coin.classList.add('blue');
}
}
setPrevData(res);
setCoinsList(res);
}
setInterval(() => {
fetchData2();
}, 10000)
}, []);
我正在尝试制作一个加密货币价格跟踪器网站。我在每 10 秒更新一次 coinList 变量 prevData 变量后获取数据。我也想访问 useEffect 中的 prevData 变量,但在它里面变量没有更新,它的值仍然是 []。但是,如果我在 useEffect 之外使用 prevData,数据就会发生变化,我可以使用它。
我只想要正在更新的 useEffect 中 prevData 的值。
function App() {
const [coinsList, setCoinsList] = useState([]);
const [prevData, setPrevData] = useState([]);
let [width, setWidth] = useState(window.innerWidth)
useEffect(() => {
async function fetchData() {
const response = await fetch('api key').then(res => res.json())
setCoinsList(res);
setPrevData(res);
}
fetchData()
async function fetchData2() {
const res = await fetch('api key').then(res => res.json())
for (let i = 0; i < prevData.length; i++) {
if (res[i].current_price > prevData[i].current_price) {
let coin = document.getElementById(`${prevData[i].id}`)
coin.classList.add('green')
console.log(coin)
// setTimeout(() => {
// coin.classList.remove('green')
// }, 1000)
} else if (res[i].current_price < prevData[i].current_price) {
let coin = document.getElementById(`${prevData[i].id}`)
coin.classList.add('red')
console.log(coin)
// setTimeout(() => {
// coin.classList.remove('red')
// }, 1000)
} else {
let coin = document.getElementById(`${prevData[i].id}`)
console.log(coin)
coin.classList.add('blue')
// setTimeout(() => {
// coin.classList.remove('blue')
// }, 1000)
}
}
setPrevData(res);
setCoinsList(res);
}
setInterval(() => {
fetchData2()
console.log(prevData) // it is not updating
}, 10000)
}, [])
useEffect(() => {
window.addEventListener('resize', () => {
setWidth(window.innerWidth)
})
}, [window.innerWidth])
console.log(prevData) // it is updating
您在 setInterval
范围内的 prevData
值上有 closure,您可以使用 ref 修复它:
const prevDataRef = useRef();
// Keep the ref updates
useEffect(() => {
prevDataRef.current = prevData;
}, []);
// no closure
useEffect(() => {
setInterval(() => {
fetchData2(); // use prevDataRef.current
console.log(prevDataRef.current); // updated
}, 10000);
}, []);
在 fetchData2
中,您有一个 prevData
状态的陈旧外壳。它在间隔回调范围内关闭。
一种补救方法是使用 React ref 和额外的 useEffect
挂钩来缓存 prevData
状态的副本,可以在 fetchData2
回调中访问。
function App() {
const [coinsList, setCoinsList] = useState([]);
const [prevData, setPrevData] = useState([]);
const prevDataRef = React.useRef(prevData);
useEffect(() => {
prevDataRef.current = prevData;
}, [prevData]);
...
useEffect(() => {
...
async function fetchData2() {
const prevData = prevDataRef.current; // <-- grab current value
const res = await fetch('api key').then(res => res.json());
for (let i = 0; i < prevData.length; i++) {
if (res[i].current_price > prevData[i].current_price) {
let coin = document.getElementById(`${prevData[i].id}`);
coin.classList.add('green');
} else if (res[i].current_price < prevData[i].current_price) {
let coin = document.getElementById(`${prevData[i].id}`);
coin.classList.add('red');
} else {
let coin = document.getElementById(`${prevData[i].id}`);
coin.classList.add('blue');
}
}
setPrevData(res);
setCoinsList(res);
}
setInterval(() => {
fetchData2();
}, 10000)
}, []);