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

vue中defineProperty和Proxy的區(qū)別詳解

 更新時間:2020年11月30日 11:13:03   作者:南藍(lán)  
這篇文章主要介紹了vue中defineProperty和Proxy的區(qū)別詳解,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧

Proxy的出現(xiàn),給vue響應(yīng)式帶來了極大的便利,比如可以直接劫持?jǐn)?shù)組、對象的改變,可以直接添加對象屬性,但是兼容性可能會有些問題

Proxy可以劫持的數(shù)組的改變,defineProperty 需要變異

defineProperty 中劫持?jǐn)?shù)組變化的變異的方法

可以理解為在數(shù)組實(shí)例和原型之間,插入了一個新的原型的對象,這個原型方法實(shí)現(xiàn)了變異的方法,也就真正地攔截了數(shù)組原型上的方法

我們來看下vue2.x的源碼

// vue 2.5.0
var arrayProto = Array.prototype;
var arrayMethods = Object.create(arrayProto); // Array {}
function def(obj, key, val, enumerable) {
  Object.defineProperty(obj, key, {
   value: val,
   enumerable: !!enumerable,
   writable: true,
   configurable: true
  });
}
var methodsToPatch = [
  'push',
  'pop',
  'shift',
  'unshift',
  'splice',
  'sort',
  'reverse'
  ];

  /**
  * Intercept mutating methods and emit events
  */
  methodsToPatch.forEach(function(method) {
  // cache original method
  var original = arrayProto[method]; 
    // 比如 method是push,則結(jié)果為
    // ƒ push() { [native code] }
  def(arrayMethods, method, function mutator() {
   var args = [],
   len = arguments.length;
   while (len--) args[len] = arguments[len];

   var result = original.apply(this, args);
   var ob = this.__ob__;
   var inserted;
   switch (method) {
   case 'push':
   case 'unshift':
    inserted = args;
    break
   case 'splice':
    inserted = args.slice(2);
    break
   }
   if (inserted) {
   ob.observeArray(inserted);
   }
   // notify change
   ob.dep.notify();
   return result
  });
  });
  
  /**
  * Observe a list of Array items.
  */
  Observer.prototype.observeArray = function observeArray(items) {
  for (var i = 0, l = items.length; i < l; i++) {
   observe(items[i]); // 后續(xù)的邏輯
  }
  };

Proxy可以直接劫持?jǐn)?shù)組的改變

  let proxy = new Proxy(fruit, {
    get: function (obj, prop) {
      return prop in obj ? obj[prop] : undefined

    },
    set: function (obj, prop, newVal) {
      obj[prop] = newVal
      console.log("newVal", newVal) // 輸出{ name: "lemon", num: 999 }
      return true;
    }
  })
  proxy.push({ name: "lemon", num: 999 })
  console.log(fruit)

Proxy代理可以劫持對象的改變,defineProperty需要遍歷

defineProperty

   let fruit = {
     "apple": 2,
     "pear": 22,
     "peach": 222
  }
  Object.keys(fruit).forEach(function (key) {
      Object.defineProperty(fruit[i], key, {
        enumerable: true,
        configurable: true,
        get: function () {
          return val;

        },
        set: function (newVal) {
          val = newVal; // 輸出 newVal 888
          console.log("newVal", newVal)
        }
      })
    })
   fruit.apple = 888

Proxy

   let fruit = {
     "apple": 2,
     "pear": 22,
     "peach": 222
  }
  let proxy = new Proxy(fruit, {
    get: function (obj, prop) {
      return prop in obj ? obj[prop] : undefined

    },
    set: function (obj, prop, newVal) {
      obj[prop] = newVal
      console.log("newVal", newVal) // 輸出 newVal 888
      return true;
    }
  })
  proxy.apple = 888

Proxy代理可以劫持對象屬性的添加,defineProperty用this.$set來實(shí)現(xiàn)
defineProperty,如果屬性不存在,則需要借助this.$set

<div id="app">
  <span v-for="(value,name) in fruit">{{name}}:{{value}}個 </span> 
  <button @click="add()">添加檸檬</button>
</div>
<script src="https://unpkg.com/vue"></script>
<script>
  new Vue({
    el: '#app',
    data() {
      return {
        fruit: {
          apple: 1, 
          banana: 4, 
          orange: 5 
        }
      }
    },

    methods: {
      add() {
          this.fruit.lemon = 5;  // 不會讓視圖發(fā)生變化
        // this.$set(this.fruit,"lemon",5)  // this.$set可以
      }
    }
  })
</script>

  Object.keys(fruit).forEach(function (key) {
    Object.defineProperty(fruit, key, {
      enumerable: true,
      configurable: true,
      get: function () {
        return val;

      },
      set: function (newVal) {
        val = newVal;
        console.log("newVal", newVal) // 根本沒有進(jìn)去這里
      }
    })
  })

Proxy 直接可以添加屬性

// vue 3
<div id="app">
  <span v-for="(value,name) in fruit">{{name}}:{{value}}個 </span>
  <button @click="add()">添加檸檬</button>
</div>
<script src="https://unpkg.com/vue@next"></script>
<script>
  Vue.createApp({
    data() {
      return {
        fruit: {
          apple: 1,
          banana: 4,
          orange: 5
        }
      }
    },

    methods: {
      add() {
        this.fruit.lemon = 5; // 這樣子是可以的
      }
    }
  }).mount('#app') // vue 3 不再是使用el屬性,而是使用mount
</script>

  let proxy = new Proxy(fruit, {
    get: function (obj, prop) {
      return prop in obj ? obj[prop] : undefined

    },
    set: function (obj, prop, newVal) {
      obj[prop] = newVal
      console.log("newVal", newVal) // lemon, 888
      return true;
    }
  })
  proxy.lemon = 888

Proxy

其他屬性

應(yīng)用場景 promisify化

用Proxy寫一個場景,請求都是通過回調(diào),如果我們需要用promise包一層的話,則可以

// server.js
// 假設(shè)這里都是回調(diào)
export const searchResultList = function (data, callback, errorCallback) {
 axios.post(url, data, callback, errorCallback)
}
// promisify.js
import * as server from './server.js'
const promisify = (name,obj) => (option) => {
 return new Promise((resolve, reject) => {
  return obj[name](
   option,
   resolve,
   reject,
  )
 })
}
const serverPromisify = new Proxy(server, {
 get (target,prop) {
  return promisify(prop, server)
 }
})
export default serverPromisify

使用

// index.js
import serverPromisify from './serverPromisify'
serverPromisify.searchResultList(data).then(res=>{

})

如有不正確,望請指出

留下一個疑問,既然兼容性不是很好,那么尤大是怎么處理polyfill呢

 到此這篇關(guān)于vue中defineProperty和Proxy的區(qū)別詳解的文章就介紹到這了,更多相關(guān)vue defineProperty和Proxy內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 基于vue中的scoped坑點(diǎn)解說

    基于vue中的scoped坑點(diǎn)解說

    這篇文章主要介紹了基于vue中的scoped坑點(diǎn)解說,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2020-09-09
  • 關(guān)于vue3.0使用axios報錯問題

    關(guān)于vue3.0使用axios報錯問題

    這篇文章主要介紹了vue3.0使用axios報錯問題記錄,vue-cli3.0安裝插件的時候要注意區(qū)分vue-cli2.0的命令,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2022-08-08
  • vue elementUI tree樹形控件獲取父節(jié)點(diǎn)ID的實(shí)例

    vue elementUI tree樹形控件獲取父節(jié)點(diǎn)ID的實(shí)例

    今天小編就為大家分享一篇vue elementUI tree樹形控件獲取父節(jié)點(diǎn)ID的實(shí)例,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2018-09-09
  • vue自定義指令用法經(jīng)典實(shí)例小結(jié)

    vue自定義指令用法經(jīng)典實(shí)例小結(jié)

    這篇文章主要介紹了vue自定義指令用法,結(jié)合實(shí)例形式總結(jié)分析了vue自定義指令常見寫法與相關(guān)操作注意事項(xiàng),需要的朋友可以參考下
    2019-03-03
  • 解決Vue中使用keepAlive不緩存問題

    解決Vue中使用keepAlive不緩存問題

    這篇文章主要介紹了Vue中使用keepAlive不緩存問題,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價值,需要的朋友可以參考下
    2020-08-08
  • 一次搞清Vue3中組件通訊的全部方式

    一次搞清Vue3中組件通訊的全部方式

    毫無疑問,組件通訊是Vue中非常重要的技術(shù)之一,它的出現(xiàn)能夠使我們非常方便的在不同組件之間進(jìn)行數(shù)據(jù)的傳遞,以達(dá)到數(shù)據(jù)交互的效果,所以,學(xué)習(xí)組件通訊技術(shù)是非常有必要的,本文將一次解決Vue3中組件通訊的全部方式,總有你想要的那一種,需要的朋友可以參考下
    2024-09-09
  • Vue實(shí)現(xiàn)快捷鍵錄入功能的示例代碼

    Vue實(shí)現(xiàn)快捷鍵錄入功能的示例代碼

    有的時候項(xiàng)目需要在頁面使用快捷鍵,而且需要對快捷鍵進(jìn)行維護(hù)。本文將為大家展示Vue實(shí)現(xiàn)快捷鍵錄入功能的示例代碼,感興趣的可以了解一下
    2022-04-04
  • Vue 列表渲染 key的原理和作用詳解

    Vue 列表渲染 key的原理和作用詳解

    這篇文章主要介紹了key在Vue列表渲染時的原理和作用,文中通過示例代碼介紹的非常詳細(xì),對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧
    2021-10-10
  • 關(guān)于vue表單提交防雙/多擊的例子

    關(guān)于vue表單提交防雙/多擊的例子

    今天小編就為大家分享一篇關(guān)于vue表單提交防雙/多擊的例子,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧
    2019-10-10
  • vue3?證件識別上傳組件封裝功能

    vue3?證件識別上傳組件封裝功能

    證件圖片識別上傳根據(jù)業(yè)務(wù)需要,經(jīng)常涉及到證件上傳,例如身份證上傳、銀行卡、營業(yè)執(zhí)照等信息,根據(jù)設(shè)計(jì)師的設(shè)計(jì),單獨(dú)封裝了一個上傳組件,這篇文章主要介紹了vue3?證件識別上傳組件封裝,需要的朋友可以參考下
    2023-05-05

最新評論