ホームページ >ウェブフロントエンド >jsチュートリアル >React の useMemo を理解する: その機能、いつ使用するか、ベスト プラクティス
React はユーザー インターフェイスを構築するための強力なライブラリですが、アプリケーションが成長するにつれて、パフォーマンスが問題になることがあります。ここで useMemo のような React フックが役に立ちます。この記事では、useMemo の機能、いつ役立つか、および useMemo を使用する際のベスト プラクティスについて詳しく説明します。避けるべき一般的な落とし穴についても説明します。
useMemo は、計算結果をメモ化できる React フックです。簡単に言えば、関数の結果を記憶し、依存関係が変更された場合にのみ再計算します。これにより、不必要な計算が防止され、パフォーマンスが向上します。
これが基本的な例です:
import React, { useMemo } from 'react'; function ExpensiveCalculation({ num }) { const result = useMemo(() => { console.log('Calculating...'); return num * 2; }, [num]); return <div>The result is {result}</div>; }
この例では、useMemo 内の関数は、num が変更された場合にのみ実行されます。 num が同じままの場合、React は計算をスキップし、以前にメモ化された結果を使用します。
useMemo を使用する主な理由は、パフォーマンスを最適化することです。 React では、コンポーネントは状態やプロパティが変更されるたびに再レンダリングされます。これにより、特に計算が複雑な場合やコンポーネント ツリーが大きい場合、高コストの計算が必要以上に頻繁に実行される可能性があります。
大規模なデータセットのフィルタリングなど、大量の計算を実行するコンポーネントがあると想像してください。 useMemo を使用しない場合、この計算はレンダリングのたびに実行されるため、アプリケーションの速度が低下する可能性があります。
import React, { useMemo } from 'react'; function ExpensiveCalculationComponent({ numbers }) { // Expensive calculation: filtering even numbers const evenNumbers = useMemo(() => { console.log('Filtering even numbers...'); return numbers.filter(num => num % 2 === 0); }, [numbers]); return ( <div> <h2>Even Numbers</h2> <ul> {evenNumbers.map((num) => ( <li key={num}>{num}</li> ))} </ul> </div> ); } // Usage const numbersArray = Array.from({ length: 100000 }, (_, i) => i + 1); export default function App() { return <ExpensiveCalculationComponent numbers={numbersArray} />; }
この例では、フィルタリング操作は計算コストが高くなります。 useMemo でラップすると、レンダリングごとではなく、数値配列が変更されたときにのみ実行されます。
レンダリングのたびに新しい配列またはオブジェクトをプロパティとして子コンポーネントに渡すと、内容が変更されていない場合でも、不必要な再レンダリングが発生する可能性があります。 useMemo を使用して配列またはオブジェクトをメモ化できます。
import React, { useMemo } from 'react'; function ChildComponent({ items }) { console.log('Child component re-rendered'); return ( <ul> {items.map((item, index) => ( <li key={index}>{item}</li> ))} </ul> ); } export default function ParentComponent() { const items = useMemo(() => ['apple', 'banana', 'cherry'], []); return ( <div> <h2>Fruit List</h2> <ChildComponent items={items} /> </div> ); }
ここでは、item 配列が useMemo を使用してメモ化されており、ChildComponent が必要な場合にのみ再レンダリングされるようにしています。 useMemo を使用しないと、レンダリングのたびに新しい配列が作成され、子コンポーネントの不必要な再レンダリングが発生します。
大規模なコンポーネント ツリーを操作する場合、useMemo を使用すると、特に深くネストされたコンポーネント内で負荷の高い操作の場合、不必要な再レンダリングを減らすことができます。
import React, { useMemo } from 'react'; function LargeComponentTree({ data }) { const processedData = useMemo(() => { console.log('Processing data for large component tree...'); return data.map(item => ({ ...item, processed: true })); }, [data]); return ( <div> <h2>Processed Data</h2> {processedData.map((item, index) => ( <div key={index}>{item.name}</div> ))} </div> ); } // Usage const largeDataSet = Array.from({ length: 1000 }, (_, i) => ({ name: `Item ${i + 1}` })); export default function App() { return <LargeComponentTree data={largeDataSet} />; }
この例では、useMemo を使用して、コンポーネントでレンダリングする前に大規模なデータセットを処理します。処理されたデータをメモ化することで、コンポーネントは元のデータプロパティが変更された場合にのみデータを再計算し、不必要な再処理を回避し、パフォーマンスを向上させます。
useMemo は強力なツールですが、正しく使用することが重要です。以下にいくつかのベストプラクティスを示します:
const expensiveCalculation = useMemo(() => { console.log('Calculating sum...'); return numbers.reduce((acc, num) => acc + num, 0) * multiplier; }, [numbers, multiplier]);
この計算は数値または乗数が変更された場合にのみ再実行され、他の再レンダリングで不要な再計算を節約できる可能性があります。
}, [numbers, multiplier]); // Correct dependencies
依存関係から乗数を省略した場合、乗数が変更されたときに計算が更新されず、不正確な結果が得られます。
const simpleValue = useMemo(() => { return 42; // This is not a complex calculation }, []); // Empty dependencies array
値は定数であり、計算は簡単であるため、このメモ化は不要です。パフォーマンス上の利点はなく、複雑さが増します。
const handleClick = () => { console.log('Button clicked'); };
この関数はシンプルで、重い計算は必要ありません。これをメモ化すると、パフォーマンスが大幅に向上することなく、コードが不必要に複雑になります。
これらのベスト プラクティスに従うことで、useMemo を効果的に使用して、コードを過度に複雑にしたり、不適切な依存関係管理による潜在的なバグを導入したりすることなく、React コンポーネントを最適化できます。
useMemo は優れたツールですが、注意すべきよくある間違いがいくつかあります。
依存関係の無視: 配列に依存関係を含めるのを忘れると、メモ化された値が古くなり、バグが発生する可能性があります。メモ化された関数内で使用されるすべての変数が依存関係配列に含まれていることを常に再確認してください。
どこでも useMemo を使用する: すべての関数や値をメモ化する必要はありません。コードにパフォーマンスの問題がない場合、useMemo を追加しても状況は改善されません。実際、メモ化のオーバーヘッドにより処理がわずかに遅くなる可能性があります。
再レンダリングの誤解: useMemo は、コンポーネントのレンダリング プロセス全体ではなく、メモ化された計算のみを最適化します。コンポーネントが新しい props または state をまだ受け取っている場合は、メモ化された値が変更されていなくても、コンポーネントは再レンダリングされます。
useMemo は React アプリケーションのパフォーマンスを最適化するための強力なフックですが、賢明に使用する必要があります。実際のパフォーマンスのボトルネックがある場所での使用に重点を置き、依存関係が正しいことを常に確認してください。これらのベスト プラクティスに従うことで、よくある落とし穴を回避し、プロジェクトで useMemo を最大限に活用できます。
以上がReact の useMemo を理解する: その機能、いつ使用するか、ベスト プラクティスの詳細内容です。詳細については、PHP 中国語 Web サイトの他の関連記事を参照してください。