Vue.$set の失敗の落とし穴の発見と解決

Vue.$set の失敗の落とし穴の発見と解決

偶然、プロジェクト内でVue.$setが無効であることがわかりました

データ フィルタリングを追加する必要があります。左側はコントロールの選択、中央は条件、右側は値です。

コントロールに応じて異なる値オプションコントロールがレンダリングされるため

    <el-form インライン>
      <el-form-item スタイル="margin-bottom: 20px;">
        <el-select v-model="data[props.prop]" @change="data[props.value] = ''">
          <el-オプション
            v-for="コントロール内の項目"
            :key="アイテムID"
            :value="アイテムID"
            :label="item.label">
          </el-option>
        </el-select>
      </el-form-item>
      <el-form-item スタイル="margin-bottom: 20px;">
        <el-select v-model="データ[props.type]">
          <el-オプション
            v-for="条件内のアイテム"
            :key="アイテムコード"
            :value="アイテムコード"
            :label="アイテム名"
          </el-option>
        </el-select>
      </el-form-item>
      <el-form-item スタイル="margin-bottom: 20px;">
        <FormControl v-if="control" :control="control" :value="data[props.value]" @input="onValueChange" ></FormControl>
        <el-input v-else :value="data[props.value]" @input="onValueChange"></el-input>
      </el-form-item>
      <el-form-item>
        <el-button type="primary" icon="el-icon-plus" @click="add"></el-button>
      </el-form-item>
    </el-form>
{
  小道具:{
    小道具: {
      タイプ: オブジェクト、
      デフォルト: () => ({
        プロップ: 'プロップ',
        値: '値',
        タイプ: 'タイプ'
      })
    }
  },
  データ(){
    戻る {
      データ:{
        
      }
    }
  },
  方法:{
    onValueChange(val){
      this.$set(this.data, this.props.value, val)
    }
  }
  //...
}

コードスニペット

コントロールIDが不確定なため、すべてのデータをキーであらかじめプリセットしておくことはできず、当然応答は不可能です。そこで、onValueChangeでthis.$setを使用して動的にデータを追加し、応答を実現しています。

再現すると、値入力ボックスのデータがリアルタイムで応答できないことがわかります

$set を使ったのですが反応しませんでした。調べてみると、コントロールを切り替えた後に値が反応しないことがわかりました。ただし、切り替える前に何か入力してから切り替えれば問題はありません。

さらに調査した結果、

<el-select v-model="data[props.prop]" @change="data[props.value] = ''">

@changeイベントを削除すると問題は解決します

問題は、data[props.value] = '' の場合に発生します。

Vueのソースコードを確認してください

//vue/src/core/observer/index.js ソースコードスニペット/**
 * オブジェクトにプロパティを設定します。新しいプロパティを追加し、
 * プロパティが
 * すでに存在します。
 */
エクスポート関数セット (ターゲット: 配列<any> | オブジェクト、キー: any、値: any): any {
  (process.env.NODE_ENV !== 'production' && の場合
    (isUndef(ターゲット) || isPrimitive(ターゲット))
  ){
    警告(`未定義、null、またはプリミティブ値にリアクティブ プロパティを設定できません: ${(target: any)}`)
  }
  Array.isArray(ターゲット) && isValidArrayIndex(キー) の場合 {
    ターゲットの長さ = Math.max(ターゲットの長さ、キー)
    ターゲット.splice(キー、1、値)
    戻り値
  }
  if (ターゲットのキー && !(Object.prototype のキー)) {
    ターゲット[キー] = 値
    戻り値
  }
  const ob = (ターゲット: 任意).__ob__
  target._isVue の場合、ob と ob.vmCount は次のように記述します。
    process.env.NODE_ENV !== 'production' && 警告(
      'Vue インスタンスまたはそのルート $data にリアクティブ プロパティを追加しないでください ' +
      「実行時 - データ オプションで事前に宣言します。」
    )
    戻り値
  }
  もし (!ob) {
    ターゲット[キー] = 値
    戻り値
  }
  defineReactive(ob.value, key, val)
  ob.dep.notify()
  戻り値
}

defineReactive の前に、オブジェクト内にキーが存在するかどうかが判断されていることがわかります。存在する場合はスキップされます。

問題はここにあります。Vue.$set ドキュメントを何度も読みましたが、$set では既存のキーに監視オブジェクトを追加できないことに気づきませんでした。

data[props.value] = ''を削除し、onValueChange('')に変更すると問題が完全に解決されます

要約する

Vue.$set の前に、オブジェクトにキーが存在してはいけません。そうでない場合は、値のみが更新され、キーの応答監視は追加されません。

上記は私の個人的な経験です。参考になれば幸いです。また、123WORDPRESS.COM を応援していただければ幸いです。

以下もご興味があるかもしれません:
  • Vue での this.$set の動的データバインディングのケーススタディ
  • Vue - 計算プロパティ「****」が割り当てられましたが、セッターがありませんというエラーの解決。
  • vue3.0 でのセットアップの使用 (2 つの使用方法)
  • Vue で resetFields() を使用する際の注意事項について簡単に説明します。
  • Vue の vue.$set() メソッドのソースコード例の詳細な説明

<<:  Oracle の開閉の 4 つのモード

>>:  Linux TTY/PTS の違いの概要

推薦する

自己終了XHTMLタグを書くときに注意すべきこと

XHTMLの img タグは、次のように記述する必要があります: <img alt="...

TypeScriptはvscodeを使用してコードのコンパイルプロセスを監視します

インストールtsコマンドをグローバルにインストールする npm install -g typescr...

Dockerコンテナ内のホストのホスト名が取得できない問題の解決方法

Node.js環境でテストが通っています。他の言語でも同様です。環境変数を取得する方法を使うだけです...

Win10 への MySQL 5.7 のインストール MSI 版のチュートリアル(画像とテキスト付き)

主にインストール後に my.ini ファイルを確認するために、msi 形式でインストールしました。フ...

W3C チュートリアル (2): W3C プログラム

W3C 標準化プロセスは 7 つの異なるステップに分かれています。 W3C 標準化プロセスは 7 つ...

Linuxのアラーム機能の例の説明

Linuxアラーム機能の紹介上記のコード: #include <stdio.h> #in...

3次元画像配置効果を実現する純粋なCSSのサンプルコード

1. 要素の幅/高さ/パディング/マージンのパーセンテージ基準要素の幅/高さ/パディング/マージンの...

フローチャートとUIフローの違い

UI デザインにおける多くの概念は言葉で言えば似ているように見えるかもしれませんが、実際には大きく異...

一般的な HTTP ステータス コード 10 個の詳細な説明

HTTP ステータス コードは、Web サーバーの HTTP 応答ステータスを示すために使用される ...

CSS3 のテキストとフォントの新しい設定

テキストシャドウテキストシャドウ: 水平オフセット 垂直オフセット ぼかし色互換性: IE10+ &...

MySQLデータ遅延ジャンプの問題の解決策

今日は、データベース遅延ジャンプに関する別の典型的な問題を分析しました。このプロセスでは、参考のため...

Vue での keepAlive の使用例の詳細な説明

開発においては、一覧から詳細ページにジャンプし、また詳細ページに戻る際に一覧ページの状態(スクロール...

Linux での MySQL データベースのマスター スレーブ同期レプリケーション構成

Linux での MySQL データベースのマスター/スレーブ同期構成の利点は、この方法をバックアッ...

Centos7 で yum を使用して Ceph 分散ストレージをインストールするチュートリアル

目次序文yumソース、epelソースを設定するCephソースの設定Cephとそのコンポーネントをイン...

Linux で実行中のすべてのプロセスを表示する方法

ps コマンドを使用できます。プロセスの PID など、現在実行中のプロセスに関する関連情報を表示で...