Rumah > Soal Jawab > teks badan
P粉4267805152023-08-21 00:02:16
Masalah yang ditunjukkan dalam kod di atas ialah selepas mendapat data, terdapat kelewatan async 500ms tambahan sebelum menetapkan keadaan. Dalam kod sebenar, ia kelihatan seperti terdapat pemprosesan tambahan (mungkin segerak) yang menyebabkan setData
在.all
dipanggil selepas.
Perkara terbaik untuk dilakukan ialah membuat done
作为一个计算属性而不是一个单独的状态,因为在那个点上,您不需要依赖于状态设置竞争,并且Object.keys(data).length
cukup murah supaya ia tidak merendahkan prestasi (dan anda menggunakannya di kawasan lain, jika ia menjadi masalah, anda boleh menyimpannya ke dalam pembolehubah).
const [data, setData] = useState({}); const done = Object.keys(data).length === 20; // 在实际代码中为200
const { useState } = React; const Example = () => { const [data, setData] = useState({}); const done = Object.keys(data).length === 20; // 在实际代码中为200 const demoData = Array.from(Array(20).keys()); const demoResolver = (x) => new Promise(res => setTimeout(() => res(x), Math.random() * 1250)) const loadData = () => { const promises = demoData.map(c => demoResolver(c)); promises.forEach(promise => { promise .then(r => { setTimeout(() => { setData(p => ({ ...p, [r]: r })); }, 500); }) }); } console.log('Render', Object.keys(data).length); const progressBarIsShownDebugColor = (done) ? 'is-danger' : 'is-info'; return ( <section className='section'> <h1 className='title is-3'>{'Example'}</h1> <progress max={demoData.length} value={Object.keys(data).length} className={'progress my-3 ' + progressBarIsShownDebugColor} /> <button onClick={loadData}>Start</button> </section> ) } ReactDOM.render(<Example />, document.getElementById("react"));
.as-console-wrapper { max-height: 50px !important; }
<script src="https://cdnjs.cloudflare.com/ajax/libs/react/17.0.1/umd/react.production.min.js"></script> <script src="https://cdnjs.cloudflare.com/ajax/libs/react-dom/17.0.1/umd/react-dom.production.min.js"></script> <link rel="stylesheet" href="https://cdn.jsdelivr.net/npm/bulma@0.9.4/css/bulma.min.css"> <div id="react"></div>