欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Vue中的this.$set()使用方法詳解(一文搞懂)

 更新時間:2025年09月06日 11:52:54   作者:前端頁面仔  
this.$set是Vue中的一個方法,用于在響應(yīng)式對象上設(shè)置新的屬性或修改已有的屬性,強(qiáng)制刷新并確保這些屬性也是響應(yīng)式的,這篇文章主要介紹了Vue中this.$set()使用方法的相關(guān)資料,需要的朋友可以參考下

前言

在 Vue.js 開發(fā)中,this.$set() 是一個解決響應(yīng)性問題的關(guān)鍵工具。本文將從基礎(chǔ)使用到高級場景,全面解析這個方法的使用技巧和最佳實(shí)踐。

為什么需要this.$set()?——響應(yīng)性原理的核心問題

Vue 的響應(yīng)性系統(tǒng)無法檢測對象屬性的添加或刪除,以及數(shù)組索引訪問的變化。這是因?yàn)?Vue 2 使用 Object.defineProperty() 實(shí)現(xiàn)響應(yīng)性,它存在以下限制:

// 對象屬性添加問題
const obj = { name: 'John' };
this.person = obj;

// 添加新屬性 - Vue 無法檢測
this.person.age = 30; // ? 非響應(yīng)式

// 數(shù)組索引修改問題
this.numbers = [1, 2, 3];
this.numbers[1] = 99; // ? 非響應(yīng)式

這些情況下,視圖不會自動更新,這時就需要 this.$set() 出馬。

基本語法和參數(shù)解析

this.$set() 的完整語法如下:

this.$set(target, propertyName/index, value)
  • ??target??:要修改的目標(biāo)對象或數(shù)組(必需)
  • ??propertyName/index??:要添加或修改的屬性名或數(shù)組索引(必需)
  • ??value??:要設(shè)置的值(必需)

使用場景和示例

場景1:為響應(yīng)式對象添加新屬性

export default {
  data() {
    return {
      user: {
        name: 'Alice',
        email: 'alice@example.com'
      }
    }
  },
  methods: {
    addUserAge() {
      // 錯誤方式 ?
      // this.user.age = 25; // 視圖不會更新
      
      // 正確方式 ?
      this.$set(this.user, 'age', 25);
      
      // 驗(yàn)證
      console.log(this.user); // 包含 age 屬性,視圖會更新
    }
  }
}

場景2:修改數(shù)組指定索引的值

export default {
  data() {
    return {
      colors: ['red', 'green', 'blue']
    }
  },
  methods: {
    updateColor(index, newColor) {
      // 錯誤方式 ?
      // this.colors[index] = newColor; // 視圖不會更新
      
      // 正確方式 ?
      this.$set(this.colors, index, newColor);
    }
  }
}

場景3:修改嵌套對象中的屬性

export default {
  data() {
    return {
      company: {
        name: 'TechCorp',
        departments: {
          engineering: {
            manager: 'John',
            size: 50
          }
        }
      }
    }
  },
  methods: {
    updateManager(name) {
      // 為嵌套對象添加新屬性
      this.$set(this.company.departments.engineering, 'manager', name);
      
      // 修改已有屬性(等效直接賦值但確保響應(yīng)性)
      this.$set(this.company.departments.engineering, 'size', 55);
    }
  }
}

原理揭秘:$set()背后的魔法

this.$set() 實(shí)際上是對全局 Vue.set() 方法的別名,其核心實(shí)現(xiàn)邏輯是:

  1. 判斷目標(biāo)是否是響應(yīng)式對象
  2. 如果對象已有該屬性:
    • 直接更新值
    • 觸發(fā)相關(guān)依賴更新
  3. 如果對象沒有該屬性:
    • 將新屬性轉(zhuǎn)為響應(yīng)式
    • 通知依賴更新
  4. 如果是數(shù)組且索引存在:
    • 使用 splice 方法修改數(shù)組
    • 觸發(fā)數(shù)組更新檢測
// 偽代碼實(shí)現(xiàn)
function set(target, key, val) {
  // 如果是數(shù)組且索引有效
  if (Array.isArray(target) && isValidArrayIndex(key)) {
    target.length = Math.max(target.length, key);
    target.splice(key, 1, val);
    return val;
  }
  
  // 如果對象已存在該屬性
  if (key in target && !(key in Object.prototype)) {
    target[key] = val;
    return val;
  }
  
  // 獲取目標(biāo)對象的觀察者實(shí)例
  const ob = target.__ob__;
  
  // 如果是非響應(yīng)式對象,直接賦值
  if (!ob) {
    target[key] = val;
    return val;
  }
  
  // 將新屬性轉(zhuǎn)為響應(yīng)式
  defineReactive(ob.value, key, val);
  
  // 通知依賴更新
  ob.dep.notify();
  return val;
}

高級應(yīng)用場景

動態(tài)表單字段管理

<template>
  <div>
    <div v-for="(field, index) in dynamicForm" :key="index">
      <input v-model="field.value" :placeholder="field.placeholder">
      <button @click="removeField(index)">Remove</button>
    </div>
    <button @click="addField">Add Field</button>
  </div>
</template>

<script>
export default {
  data() {
    return {
      dynamicForm: []
    }
  },
  methods: {
    addField() {
      const newIndex = this.dynamicForm.length;
      
      // 使用 $set 確保新字段響應(yīng)式
      this.$set(this.dynamicForm, newIndex, {
        value: '',
        placeholder: `Field ${newIndex + 1}`
      });
    },
    removeField(index) {
      // 使用 splice 確保響應(yīng)式
      this.dynamicForm.splice(index, 1);
    }
  }
}
</script>

基于權(quán)限的動態(tài)數(shù)據(jù)展示

<template>
  <div>
    <div v-if="user.permissions.viewSalary">
      <p>Salary: {{ user.salary }}</p>
    </div>
    <button @click="grantSalaryAccess">Grant Salary Access</button>
  </div>
</template>

<script>
export default {
  data() {
    return {
      user: {
        name: 'Bob',
        permissions: {
          viewSalary: false
        }
      }
    }
  },
  methods: {
    grantSalaryAccess() {
      // 1. 添加權(quán)限屬性
      this.$set(this.user.permissions, 'viewSalary', true);
      
      // 2. 添加工資字段(不會暴露給沒有權(quán)限的用戶)
      setTimeout(() => {
        // 使用 $set 確保響應(yīng)式
        this.$set(this.user, 'salary', 8500);
      }, 1000);
    }
  }
}
</script>

最佳實(shí)踐與注意事項

  1. ??避免不必要的使用??:

    • 已有屬性可以直接修改:this.existingProp = newValue
    • 數(shù)組元素修改盡量使用 push()pop()shift()unshift()splice() 等方法
  2. ??性能考慮??:

    • 對大型對象深度使用 $set 可能有性能開銷
    • 批量更新時考慮使用 Object.assign() 創(chuàng)建新對象
  3. ??替代方案??:

    // 使用新對象替換舊對象
    this.user = {
      ...this.user,
      age: 30,
      title: 'Senior Developer'
    };
    
    // 對于數(shù)組
    this.colors = this.colors.map((color, index) => 
      index === 1 ? 'purple' : color
    );
  4. ??Vue 3 的變化??:

    • Vue 3 使用 Proxy 解決響應(yīng)性限制
    • $set 在 Vue 3 中主要為兼容性保留
    • 在 Vue 3 中可直接添加新屬性:this.newProperty = value

常見問題解決方案

問題:使用$set后視圖仍未更新?

??解決方案??:

  1. 確保在 Vue 實(shí)例方法中使用(生命周期鉤子、方法等)
  2. 檢查目標(biāo)對象是否在 Vue 的響應(yīng)系統(tǒng)中(data 返回或 Vue.observable 創(chuàng)建)
  3. 使用 this.$forceUpdate() 作為最后手段(不推薦)

問題:深度嵌套對象如何處理?

??解決方案??:

// 使用自定義工具方法
setNestedProperty(obj, path, value) {
  const keys = path.split('.');
  const lastKey = keys.pop();
  let current = obj;
  
  keys.forEach(key => {
    if (!current[key]) this.$set(current, key, {});
    current = current[key];
  });
  
  this.$set(current, lastKey, value);
}

// 使用示例
this.setNestedProperty(this.app, 'settings.theme.color', 'dark-blue');

總結(jié)

this.$set() 是 Vue 響應(yīng)式系統(tǒng)的關(guān)鍵補(bǔ)充工具,尤其適用于:

  • 動態(tài)添加新屬性到已存在對象
  • 通過索引修改數(shù)組元素
  • 確保深度嵌套屬性的響應(yīng)式更新

掌握 this.$set() 的使用場景和替代方案,能幫助開發(fā)者更高效地構(gòu)建響應(yīng)式 Vue 應(yīng)用,避免常見的響應(yīng)性問題。在 Vue 3 中,由于 Proxy 的引入,大部分場景不再需要 $set,但對于 Vue 2 項目,這仍是必備工具。

到此這篇關(guān)于Vue中this.$set()使用方法的文章就介紹到這了,更多相關(guān)Vue this.$set()使用方法內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 使用Vue3和p5.js實(shí)現(xiàn)交互式圖像動畫

    使用Vue3和p5.js實(shí)現(xiàn)交互式圖像動畫

    這篇文章主要介紹了如何用Vue3和p5.js打造一個交互式圖像動畫,文中給出了詳細(xì)的代碼示例,本代碼適用于需要在網(wǎng)頁中實(shí)現(xiàn)圖像滑動效果的場景,例如圖片瀏覽、相冊展示等,感興趣的小伙伴跟著小編一起來看看吧
    2024-06-06
  • vscode vue 文件模板的配置方法

    vscode vue 文件模板的配置方法

    這篇文章主要介紹了vscode vue 文件模板的配置方法,本文給大家介紹的非常詳細(xì),具有一定的參考借鑒價值,需要的朋友可以參考下
    2019-07-07
  • vue 中使用 vxe-table 制作可編輯表格的使用過程

    vue 中使用 vxe-table 制作可編輯表格的使用過程

    這篇文章主要介紹了vue 中使用 vxe-table 制作可編輯表格的使用過程,本文通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2021-08-08
  • vue3自定義組件之v-model實(shí)現(xiàn)父子組件雙向綁定

    vue3自定義組件之v-model實(shí)現(xiàn)父子組件雙向綁定

    這篇文章主要介紹了vue3自定義組件之v-model實(shí)現(xiàn)父子組件雙向綁定方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-08-08
  • 使用Vue調(diào)取接口,并渲染數(shù)據(jù)的示例代碼

    使用Vue調(diào)取接口,并渲染數(shù)據(jù)的示例代碼

    今天小編就為大家分享一篇使用Vue調(diào)取接口,并渲染數(shù)據(jù)的示例代碼,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2019-10-10
  • vue離開頁面時如何銷毀定時器

    vue離開頁面時如何銷毀定時器

    這篇文章主要介紹了vue離開頁面時如何銷毀定時器,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-05-05
  • Vue腳手架編寫試卷頁面功能

    Vue腳手架編寫試卷頁面功能

    腳手架是一種用于快速開發(fā)Vue項目的系統(tǒng)架構(gòu),這篇文章主要介紹了Vue腳手架實(shí)現(xiàn)試卷頁面功能,通過實(shí)例代碼給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-03-03
  • vue打包生成的文件的js文件過大的優(yōu)化方式

    vue打包生成的文件的js文件過大的優(yōu)化方式

    這篇文章主要介紹了vue打包生成的文件的js文件過大的優(yōu)化方式,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-04-04
  • vue3之Suspense加載異步數(shù)據(jù)的使用

    vue3之Suspense加載異步數(shù)據(jù)的使用

    本文主要介紹了vue3之Suspense加載異步數(shù)據(jù)的使用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2023-02-02
  • vue實(shí)現(xiàn)移動端觸屏拖拽功能

    vue實(shí)現(xiàn)移動端觸屏拖拽功能

    這篇文章主要為大家詳細(xì)介紹了vue實(shí)現(xiàn)移動端觸屏拖拽功能,文中示例代碼介紹的非常詳細(xì),具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2020-08-08

最新評論