亚洲乱码中文字幕综合,中国熟女仑乱hd,亚洲精品乱拍国产一区二区三区,一本大道卡一卡二卡三乱码全集资源,又粗又黄又硬又爽的免费视频

你知道Vue中神奇的$set是如何實現(xiàn)的嗎?

 更新時間:2022年12月13日 15:43:13   作者:清風丶  
在日常開發(fā)中,$set的也是一個非常實用的API。但是我們知其然更要知其所以然,接下來就跟隨小編一起看一下Vue中的$set是如何實現(xiàn)的吧

前言

在日常開發(fā)中,$set的也是一個非常實用的API,因為Vue2實現(xiàn)響應式的核心是利用了ES5的Object.defineProperty,當我們通過直接修改數(shù)組下標更改數(shù)組或者給對象添加新的屬性,這個時候Object.defineproperty是監(jiān)聽不到數(shù)據(jù)的變化的,這時候大家就會用上$set,讓修改的操作也實現(xiàn)響應,我們知其然更要知其所以然,接下來看一下Vue中的$set是如何實現(xiàn)的

應用場景

 let dataArr = ["item1"];
 let dataObject = {
      name: "ccs"
    };

    dataArr[2] = "item2";
    dataObject.age = 22;
//    響應失敗,頁面沒有顯示更新新增的數(shù)據(jù)

    this.$set(this.dataArr,2,'item2')
    this.$set(this.dataObject,'age',22)
//響應成功,頁面顯示更新新增的數(shù)據(jù)

set實現(xiàn)

接下來我們看一下$set在Vue中的定義

function set(target: Array<any> | Object, key: any, val: any): any {
  if (
    process.env.NODE_ENV !== "production" &&
    (isUndef(target) || isPrimitive(target))
  ) {
    warn(
      `Cannot set reactive property on undefined, null, or primitive value: ${(target: any)}`
    );
  }
  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;
  }
  const ob = (target: any).__ob__;
  if (target._isVue || (ob && ob.vmCount)) {
    process.env.NODE_ENV !== "production" &&
      warn(
        "Avoid adding reactive properties to a Vue instance or its root $data " +
          "at runtime - declare it upfront in the data option."
      );
    return val;
  }
  if (!ob) {
    target[key] = val;
    return val;
  }
  defineReactive(ob.value, key, val);
  ob.dep.notify();
  return val;
}

在源碼中首先判斷set的目標是否是undefined和基本類型如果是undefined或基本類型就報錯,因為用戶不應該往undefined和基本類型中set東西,然后又判斷了目標是否是數(shù)組與key是不是合法的index,合法的index是指值為大于等于0的整數(shù),如果兩個條件都成立就對目標數(shù)組調(diào)用splice方法插入或者修改數(shù)組,這里的splice不是普通的splice,是王維詩里的splice,是被vue代理重寫過的splice

數(shù)組實現(xiàn)響應

$set實現(xiàn)數(shù)組修改響應的方式是代理重寫的數(shù)組的一部分方法,接下來我們看一下具體實現(xiàn)

const arrayProto = Array.prototype
export const arrayMethods = Object.create(arrayProto)

const methodsToPatch = [
  'push',
  'pop',
  'shift',
  'unshift',
  'splice',
  'sort',
  'reverse'
]
function def(obj, key, val, enumerable) {
    Object.defineProperty(obj, key, {
        value: val,
        enumerable: !!enumerable,
        writable: true,
        configurable: true
    });
}
methodsToPatch.forEach(function (method) {
  const original = arrayProto[method]
  def(arrayMethods, method, function mutator (...args) {
    const result = original.apply(this, args)
    const ob = this.__ob__
    let inserted
    switch (method) {
      case 'push':
      case 'unshift':
        inserted = args
        break
      case 'splice':
        inserted = args.slice(2)
        break
    }
    if (inserted) ob.observeArray(inserted)
    ob.dep.notify()
    return result
  })
})

vue中代理重寫的不只是splice,有push、pop、shift、unshift、splice、sort、reverse這七個方法, 首先執(zhí)行了const result = original.apply(this, args)執(zhí)行原本數(shù)組的方法并獲取它的值,接下來判斷如果是往數(shù)組中添加值就將新添加的值也實現(xiàn)響應式,最后一步拿到這個數(shù)組的_ob_對象_ob_里的dep進行派發(fā)更新。

想深入了解vue的響應式可以查閱往期文章面試官問你Vue2的響應式原理,你怎么答?

對象實現(xiàn)響應

$set中下半部分的邏輯就是用來處理對象響應的,我們接著往下看

  if (key in target && !(key in Object.prototype)) {
    target[key] = val;
    return val;
  }
  const ob = (target: any).__ob__;
  if (!ob) {
    target[key] = val;
    return val;
  }
  defineReactive(ob.value, key, val);
  ob.dep.notify();
  return val;

首先判斷了屬性如果在目標對象中直接return結(jié)束邏輯,因為vue只有添加目標對象中原本沒有的屬性時才會失去響應

例如 let obj={} obj.name='ccs',

vue在初始化的時候會將data里的所有屬性都變成響應式,如果的值是對象或者數(shù)組則會new一個Observer實例儲存在__ob__,想深入了解vue的響應式可以查閱往期文章面試官問你Vue2的響應式原理,你怎么答

拿到這個對象的_ob_進行判斷,如果不存在就說明是未經(jīng)過vue初始化的普通對象而不是響應式對象 否則就手動通過defineReactive為屬性添加get方法與set方法實現(xiàn)響應,然后手動調(diào)用dep里的notify()發(fā)布更新。

總結(jié)

vue中$set方法對數(shù)組和對象的處理本質(zhì)上的一樣的,對新增的值添加響應然后手動觸發(fā)派發(fā)更新。

以上就是你知道Vue中神奇的$set是如何實現(xiàn)的嗎?的詳細內(nèi)容,更多關(guān)于Vue實現(xiàn)$set的資料請關(guān)注腳本之家其它相關(guān)文章!

相關(guān)文章

  • vue中實現(xiàn)支持txt,docx,xlsx,mp4格式文件預覽功能(純前端)

    vue中實現(xiàn)支持txt,docx,xlsx,mp4格式文件預覽功能(純前端)

    對于Vue你可以實現(xiàn)文件的預覽功能,這篇文章主要給大家介紹了關(guān)于vue中實現(xiàn)支持txt,docx,xlsx,mp4格式文件預覽功能的相關(guān)資料,文中通過代碼介紹的非常詳細,需要的朋友可以參考下
    2023-11-11
  • vue3實現(xiàn)動態(tài)側(cè)邊菜單欄的幾種方式簡單總結(jié)

    vue3實現(xiàn)動態(tài)側(cè)邊菜單欄的幾種方式簡單總結(jié)

    在做開發(fā)中都會遇到的需求,每個用戶的權(quán)限是不一樣的,那他可以訪問的頁面(路由)可以操作的菜單選項是不一樣的,如果由后端控制,我們前端需要去實現(xiàn)動態(tài)路由,動態(tài)渲染側(cè)邊菜單欄,這篇文章主要給大家介紹了關(guān)于vue3實現(xiàn)動態(tài)側(cè)邊菜單欄的幾種方式,需要的朋友可以參考下
    2024-02-02
  • Vuex的實戰(zhàn)使用詳解

    Vuex的實戰(zhàn)使用詳解

    這篇文章主要介紹了Vuex的實戰(zhàn)使用詳解,文中通過示例代碼介紹的非常詳細,對大家的學習或者工作具有一定的參考學習價值,需要的朋友們下面隨著小編來一起學習學習吧
    2019-10-10
  • vue實現(xiàn)的上傳圖片到數(shù)據(jù)庫并顯示到頁面功能示例

    vue實現(xiàn)的上傳圖片到數(shù)據(jù)庫并顯示到頁面功能示例

    這篇文章主要介紹了vue實現(xiàn)的上傳圖片到數(shù)據(jù)庫并顯示到頁面功能,結(jié)合實例形式分析了基于vue.js的數(shù)據(jù)庫操作及頁面圖片顯示相關(guān)操作技巧,需要的朋友可以參考下
    2018-03-03
  • 使用element-ui實現(xiàn)行合并過程

    使用element-ui實現(xiàn)行合并過程

    這篇文章主要介紹了使用element-ui實現(xiàn)行合并過程,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-08-08
  • Vue項目npm操作npm run serve或npm run dev報錯及二者的區(qū)別

    Vue項目npm操作npm run serve或npm run dev報錯及二者

    這篇文章主要介紹了Vue項目npm操作npm run serve或npm run dev報錯及二者的區(qū)別說明,具有很好的參考價值,希望對大家有所幫助,如有錯誤或未考慮完全的地方,望不吝賜教
    2023-10-10
  • vue時間組件DatePicker組件的手寫示例

    vue時間組件DatePicker組件的手寫示例

    這篇文章主要為大家介紹了vue時間組件DatePicker組件的手寫實現(xiàn)示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進步,早日升職加薪
    2022-07-07
  • VUE使用draggable實現(xiàn)組件拖拽

    VUE使用draggable實現(xiàn)組件拖拽

    這篇文章主要為大家詳細介紹了VUE使用draggable實現(xiàn)組件拖拽,文中示例代碼介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們可以參考一下
    2022-04-04
  • vue按需加載實例詳解

    vue按需加載實例詳解

    在本篇文章里小編給大家整理的是關(guān)于vue按需加載實例的相關(guān)知識點內(nèi)容,有需要的朋友們可以學習參考下。
    2019-09-09
  • vue3.0語法糖內(nèi)的defineProps及defineEmits解析

    vue3.0語法糖內(nèi)的defineProps及defineEmits解析

    這篇文章主要介紹了vue3.0語法糖內(nèi)的defineProps及defineEmits解析,具有很好的參考價值,希望對大家有所幫助。如有錯誤或未考慮完全的地方,望不吝賜教
    2022-04-04

最新評論