ReactのuseEffectクロージャの落とし穴についての簡単な説明

ReactのuseEffectクロージャの落とし穴についての簡単な説明

問題コード

useEffectによって発生したクロージャの問題コードを見てみましょう

定数 btn = useRef();
const [v, setV] = useState('');

使用効果(() => {
    クリックハンドルを () とします => {
        コンソールにログ出力します。
    }
    btn.current.addEventListener('click', clickHandle)
    
    戻り値 () => {
        btn.removeEventListener('click', clickHandle)
    }
}, []);
    
定数入力ハンドル = e => {
    setV(e.target.value)
}

戻る (
        <>
            <入力値={v} onChange={inputHandle} />
            <button ref={btn} >テスト</button>
        </>
    )

useEffect の依存関係配列は空なので、内部コードはページのレンダリングが完了した後に 1 回だけ実行され、ページが破棄されたときにもう一度実行されます。このとき、入力ボックスに任意の文字を入力してテスト ボタンをクリックすると、出力は空になります。その後、どのような文字を入力して再度テスト ボタンをクリックしても、出力結果は空のままです。

なぜこのようなことが起こるのでしょうか?実際、それは閉鎖によって引き起こされます。

原因

関数のスコープは、関数が定義されるときに決定されます。

btn のクリック イベントを登録する場合、スコープは次のようになります。

現時点では、アクセス可能な自由変数 v はまだ null です。クリックイベントがトリガーされると、クリックコールバック関数が実行されます。このとき、まず実行コンテキストが作成され、スコープチェーンが実行コンテキストにコピーされます。

  • 入力ボックスに文字が入力されていない場合、クリックしたときに表示されるvは元のvのままです。
  • 入力ボックスに文字が入力されると、setV が呼び出されて状態が変更され、ページがレンダリングをトリガーし、コンポーネントの内部コードが再実行されて v が再宣言され、v は元の v ではなくなります。クリック イベントのスコープ内の v は、古い v のままです。これらは 2 つの異なる v です。

シーンを生成する

  • イベントバインディング。たとえば、サンプル コードでは、ページの初期レンダリングが完了した後にイベントが 1 回だけバインドされます。たとえば、echarts を使用する場合、useEffect で echarts のインスタンスが取得され、イベントがバインドされます。
  • タイマー。ページが読み込まれた後にタイマーを登録すると、タイマー内の関数で同様のクローズの問題も発生します。

解決

この閉鎖問題に対する 5 つの解決策は次のとおりです。

1. 代入によってvを直接変更し、vを変更するメソッドをuseCallbackでラップする

v を変更するメソッドを useCallback でラップします。useCallback でラップされた関数はキャッシュされます。依存関係の配列が空なので、ここで直接割り当てによって変更された v は古い v です。setState は状態を変更する公式に推奨される方法であるため、この方法は推奨されません。ここでは、setV は再レンダリングをトリガーするためだけに使用されています。

// 直接変更しやすいように、v の宣言を const から var に変更します。var [v, setV] = useState('');

const inputHandle = useCallback(e => {
    {値} = e.targetとする
    v = 値
    setV(値)
}, [])

2. useEffectの依存関係にvを追加する

これは、ほとんどの人が最初に思いつく解決策かもしれません。v は古いので、v が更新されるたびにイベントを再登録すればいいのではないでしょうか。しかし、これでは v が更新されるたびに再登録が必要になります。理論上は、1 回だけ登録すればよいイベントが複数回登録されることになります。

3. vの再宣言を避ける

let または var を使用して v の代わりに変数を宣言し、setState 関連関数を使用してレンダリングをトリガーする代わりに、変数を直接変更します。この方法では、変数は再宣言されず、クリック コールバック関数で「最新の」値を取得できます。ただし、この方法は推奨されません。この例では、再レンダリングがないため、入力コンポーネントには常に空の値が表示され、期待される操作を満たしていません。

4. useStateの代わりにuseRefを使用する

定数 btn = useRef();
定数vRef = useRef('');
定数[v, setV] = useStat('');

使用効果(() => {
    クリックハンドルを () とします => {
        console.log('v:', vRef.current);
    }
    btn.current.addEventListener('click', clickHandle)
    
    戻り値 () => {
        btn.removeEventListener('click', clickHandle)
    }
}, []);

定数入力ハンドル = e => {
    {値} = e.targetとする
    vRef.current = 値
    setV(値)
}

戻る (
        <>
            <入力値={v} onChange={inputHandle} />
            <button ref={btn} >テスト</button>
        </>
    )

useRef ソリューションが効果的な理由は、入力の変更ごとに vRef オブジェクトの現在のプロパティが変更され、vRef は再レンダリングされても常にその vRef であるためです。vRef はオブジェクトであるため、スタック メモリに格納されている変数の値はヒープ メモリ内のオブジェクトのアドレスであり、単なる参照です。オブジェクトの特定のプロパティのみが変更され、参照は変更されません。クリックイベントのスコープチェーンは常に同じvRefにアクセスします

5. vをオブジェクト型に置き換える

実際、useRef を使用する場合と同様に、オブジェクトである限り、特定の属性のみを変更しても、状態が指すアドレスは変更されません。

コードアドレス

テストコードを見るにはここをクリックしてください

これで、React useEffect クロージャの落とし穴に関するこの記事は終わりです。React useEffect クロージャに関するより関連性の高いコンテンツについては、123WORDPRESS.COM の過去の記事を検索するか、以下の関連記事を引き続き参照してください。今後とも 123WORDPRESS.COM をよろしくお願いいたします。

以下もご興味があるかもしれません:
  • React useEffect の理解と使用
  • React における useEffect と useLayoutEffect の違い

<<:  Linux が Sudo 権限昇格の脆弱性を公開、どのユーザーでも root コマンドを実行可能

>>:  Dockerでパラメータ変数を外部から指定する方法

推薦する

CSS3 シンプルカットカルーセル画像実装コード

実装のアイデアまず、親コンテナーを作成し、2 つの順序なしリストを使用して、柔軟なレイアウトで親コン...

Ubuntu 16.04 に Docker と nvidia-docker をインストールするための詳細なチュートリアル

目次DockerのインストールNvidia-docker のインストールDockerのインストール1...

新しい CSS display:box プロパティの詳細な説明

1. ディスプレイボックス;要素にこのプロパティを設定すると、display:inline-bloc...

Docker用国産イメージウェアハウスの使い方

1. 問題の説明何らかの理由により、中国でのDockerイメージのダウンロード速度は特に遅くなります...

MySQL マスタースレーブ遅延問題の解決方法

今日は、マスタースレーブ遅延が発生する理由とその対処方法について説明します。しっかり座って出発の準備...

Vue3 トランジションアニメーションの落とし穴記録について

目次背景問題の場所さらなる分析要約する背景私のコース「Vue 3 エンタープライズレベルの音楽アプリ...

Linux にソフトウェアをインストールするときにソフトウェア パッケージが存在しない問題を解決する方法

ソフトウェア パッケージが存在しない場合は、インストールされているソフトウェアのソフトウェア ソース...

Docker-compose におけるdepends_on 順序問題を解決する方法についての簡単な説明

コンテナをソートするためにdepends_onを使用しても、コンテナ間の依存関係の問題は完全には解決...

シンプルなタブバー切り替えコンテンツバーを実装するJavaScript

この記事では、タブバーの切り替えコンテンツバーを簡単に実現するためのJavaScriptの具体的なコ...

MySQL で誕生日から年齢を計算する複数の方法

以前はMySQLをあまり使用していなかったため、MySQLの機能にあまり詳しくありませんでした。この...

React のネストされたコンポーネントの構築順序

目次Reactの公式サイトではライフサイクルの説明を見ることができます次に、ネストされたコンポーネン...

Linux での SSH 非秘密通信の実装

SSHとは何か管理者はリモートでログインして、インターネット経由で接続されたさまざまな場所にある複数...

SQL における参照整合性の詳細な説明 (1 対 1、1 対多、多対多)

1. 参照整合性参照整合性とは、主に外部キー制約を使用した複数のテーブル間の設計を指します。複数テ...

Nginx の負荷分散アルゴリズムとフェイルオーバー分析

概要Nginx ロード バランシングは、アップストリーム サーバー (実際のビジネス ロジックによっ...

Ubuntu 16.04 64ビット版を3つのステップで32ビットプログラムと互換性を持たせる

ステップ1: システムのアーキテクチャを確認する dpkg --print-architecture...