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

詳解Vue響應(yīng)式的部分實(shí)現(xiàn)

 更新時(shí)間:2022年12月08日 14:32:48   作者:小婉子啊  
響應(yīng)式,簡(jiǎn)單來(lái)說(shuō)當(dāng)數(shù)據(jù)發(fā)生變化時(shí),對(duì)數(shù)據(jù)有依賴的代碼會(huì)重新執(zhí)行。這篇文章主要為大家介紹了Vue中響應(yīng)式的部分實(shí)現(xiàn),感興趣的可以了解一下

什么是響應(yīng)式

簡(jiǎn)單來(lái)說(shuō)當(dāng)數(shù)據(jù)發(fā)生變化時(shí),對(duì)數(shù)據(jù)有依賴的代碼會(huì)重新執(zhí)行。

例如在Vue中,當(dāng)我們的數(shù)據(jù)發(fā)生改變,界面上對(duì)該數(shù)據(jù)的引用組件會(huì)重新渲染

組件data的數(shù)據(jù)一旦變化,立即出發(fā)視圖的更新;

computed屬性在依賴發(fā)生變化時(shí),自動(dòng)重新計(jì)算新值;提供watch監(jiān)聽(tīng)器,可以監(jiān)聽(tīng)到數(shù)據(jù)的變化

Vue2與Vue3響應(yīng)式之間的區(qū)別

  • Vue2使用ES5的defineProperty實(shí)現(xiàn)
  • Vue3使用的是ES6的propxy.(PS:這也就是為什么Vue2不支持IE7/8,而Vue3不支持IE11.)

使用Object.defineProperty監(jiān)聽(tīng)對(duì)象

該方法允許精確地添加或修改對(duì)象的屬性,并返回此對(duì)象。

Object.defineProperty()方法會(huì)在直接在一個(gè)對(duì)象上定義一個(gè)新屬性,或者修改一個(gè)對(duì)象的現(xiàn)有屬性,并返回此對(duì)象

備注:(應(yīng)當(dāng)直接在object構(gòu)造器對(duì)象上調(diào)用此方法,而不是在任意一個(gè)object類型的實(shí)例上調(diào)用)

語(yǔ)法:Object.defineProperty(obj, prop, descriptor)

  • obj:要設(shè)置屬性的對(duì)象;
  • prop:要設(shè)置的屬性名,這個(gè)屬性可以是已存在也可以是不存在的;
  • descriptor:要定義或修改的屬性描述符。該參數(shù)接收一個(gè)對(duì)象,用來(lái)對(duì)屬性進(jìn)行描述。如value(值),writable(是否可重寫),enumerable(是否可枚舉)等

枚舉時(shí)使用for...inObject.keys方法可以改變這些屬性的值,默認(rèn)情況下,使用 Object.defineProperty() 添加的屬性值是不可修改(immutable)的。

對(duì)象里目前存在的屬性描述符有兩種主要形式:數(shù)據(jù)描述符和存取描述符

  • 數(shù)據(jù)描述符:是一個(gè)具有值的屬性,該值是可寫的,也可以是不可寫的
  • 存取描述符:由getter函數(shù)和setter函數(shù)所描述的屬性

一個(gè)描述符只能是這兩者其中之一,不能同時(shí)是兩者

使用Object.defineProperty監(jiān)聽(tīng)對(duì)象

利用 Object.defineProperty 重寫 getset,將對(duì)象屬性的賦值和獲取變成函數(shù),我們可以實(shí)現(xiàn)一個(gè)簡(jiǎn)單的雙向綁定

  • get: 屬性的 getter 函數(shù),如果沒(méi)有 getter,則為 undefined。當(dāng)訪問(wèn)該屬性時(shí),會(huì)調(diào)用此函數(shù)。該函數(shù)的返回值會(huì)被用作屬性的值。默認(rèn)為 undefined。
  • set: 屬性的 setter 函數(shù),如果沒(méi)有 setter,則為 undefined。當(dāng)屬性值被修改時(shí),會(huì)調(diào)用此函數(shù)。默認(rèn)為 undefined。
//實(shí)現(xiàn)一個(gè)簡(jiǎn)單的雙向綁定
const data = {}
const name = 'xiaowanzi'
Object.defineProperty(data, 'name', {
  get: function () {
    console.log('get')
    return name
  },
  set: function (newVal) {
    console.log('set')
    name = newVal
  }
})
//測(cè)試
console.log(data.name)//get xiaowanzi
data.name = 'list'//set

如果我們想讓對(duì)象的所有屬性都具有響應(yīng)式,就需要對(duì)全部屬性進(jìn)行遍歷,實(shí)現(xiàn)getter和setter:

//實(shí)現(xiàn)Vue響應(yīng)式原理
let obj = {
  name: 'aaa',
  age: 18
}
//獲取obj對(duì)象的所有key
const keys = Object.keys(obj)//Object.keys()返回一個(gè)由一個(gè)給定對(duì)象的資深可枚舉屬性組成的數(shù)組,數(shù)組中的屬性名的排列順序和正常循環(huán)遍歷該對(duì)象時(shí)返回的順序一致
//遍歷Key數(shù)組,對(duì)obj對(duì)象的每一個(gè)屬性進(jìn)行處理
keys.forEach(key => {
  //使用value變量保存key對(duì)應(yīng)的屬性值
  let value = obj[key]
  //使用Object.defineProperty
  Object.defineProperty(obj, key, {
    get() {//當(dāng)獲取屬性時(shí),回來(lái)到這里
      console.log(`${key}屬性被獲取`)
      return value
    },
    set(newValue) {//當(dāng)修改屬性時(shí),會(huì)來(lái)到這里,并且設(shè)置的值會(huì)傳給newValue
      console.log(`${key}屬性被修改`)
      //這里不能寫成obj[key]=newValue
      //如果這樣寫相當(dāng)于又對(duì)該屬性進(jìn)行修改值,又會(huì)進(jìn)入set,就死循環(huán)了
      value = newValue
    }
  })
})
?
//現(xiàn)在我們已經(jīng)可以實(shí)現(xiàn)監(jiān)聽(tīng)obj對(duì)象的讀取與修改了
console.log(obj.name)//在打印'aaa'之前會(huì)先打印'name被獲取',也就是說(shuō)監(jiān)聽(tīng)到屬性的獲取。
obj.name = 'bbb'//打印name屬性被修改,也就是說(shuō)監(jiān)聽(tīng)到了屬性的改變
?
//實(shí)現(xiàn)Vue響應(yīng)式原理
let obj={
  name:'aaa',
  age:18
}
//獲取obj對(duì)象的所有key
const keys=object.keys(obj)//Object.keys()返回一個(gè)由一個(gè)給定對(duì)象的資深可枚舉屬性組成的數(shù)組,數(shù)組中的屬性名的排列順序和正常循環(huán)遍歷該對(duì)象時(shí)返回的順序一致
//遍歷Key數(shù)組,對(duì)obj對(duì)象的每一個(gè)屬性進(jìn)行處理
keys.forEach(key=>{
  //使用value變量保存key對(duì)應(yīng)的屬性值
  let value = obj[key]
  //使用Object.defineProperty
  Object.defineProperty(obj,key,{
      get(){//當(dāng)獲取屬性時(shí),回來(lái)到這里
        console.log(`${key}屬性被獲取`)
        return value
      },
      set(newValue){//當(dāng)修改屬性時(shí),會(huì)來(lái)到這里,并且設(shè)置的值會(huì)傳給newValue
        console.log(`${key}屬性被修改`)
        //這里不能寫成obj[key]=newValue
        //如果這樣寫相當(dāng)于又對(duì)該屬性進(jìn)行修改值,又會(huì)進(jìn)入set,就死循環(huán)了
        value=newValue
      }
  })
})
?
//現(xiàn)在我們已經(jīng)可以實(shí)現(xiàn)監(jiān)聽(tīng)obj對(duì)象的讀取與修改了
console.log(obj.name)//在打印'aaa'之前會(huì)先打印'name被獲取',也就是說(shuō)監(jiān)聽(tīng)到屬性的獲取。
obj.name='bbb'//打印name屬性被修改,也就是說(shuō)監(jiān)聽(tīng)到了屬性的改變

缺點(diǎn)

可以實(shí)現(xiàn)監(jiān)聽(tīng)對(duì)象的屬性,但是它沒(méi)有辦法做到對(duì)對(duì)象新增的屬性進(jìn)行監(jiān)聽(tīng),同時(shí)也沒(méi)有辦法做到對(duì)數(shù)據(jù)進(jìn)行監(jiān)聽(tīng)

使用ES6的Proxy實(shí)現(xiàn)監(jiān)聽(tīng)對(duì)象

該API就是用來(lái)實(shí)現(xiàn)監(jiān)聽(tīng)對(duì)象的,而且該API對(duì)數(shù)組同樣也是有效果的,在使用Proxy時(shí),通常會(huì)搭配Reflect一起使用 Proxy

用于創(chuàng)建代理對(duì)象,從而實(shí)現(xiàn)基本操作的攔截和自定義(如屬性的查找,賦值,枚舉,函數(shù)調(diào)用等)

術(shù)語(yǔ):

  • handler:包含捕捉器(trap)的占位符對(duì)象,可譯為處理器對(duì)象。
  • traps:提供屬性訪問(wèn)的方法。這類似于操作系統(tǒng)中捕獲器的概念。
  • target:被 Proxy 代理虛擬化的對(duì)象。它常被作為代理的存儲(chǔ)后端。根據(jù)目標(biāo)驗(yàn)證關(guān)于對(duì)象不可擴(kuò)展性或不可配置屬性的不變量(保持不變的語(yǔ)義)

語(yǔ)法:

const p = new Proxy(target, handler)

1.第一個(gè)參數(shù)target:要包裝的目標(biāo)對(duì)象

2.第二個(gè)參數(shù)handle:接收一個(gè)對(duì)象,內(nèi)部定義了操作目標(biāo)對(duì)象時(shí)的方法;

參數(shù):

  • target:要使用 Proxy 包裝的目標(biāo)對(duì)象(可以是任何類型的對(duì)象,包括原生數(shù)組,函數(shù),甚至另一個(gè)代理)。
  • handler:一個(gè)通常以函數(shù)作為屬性的對(duì)象,各屬性中的函數(shù)分別定義了在執(zhí)行各種操作時(shí)代理 p 的行為。

方法

Reflect

是一個(gè)內(nèi)置的對(duì)象,它提供攔截 JavaScript 操作的方法。Reflect不是一個(gè)函數(shù)對(duì)象,因此它是不可構(gòu)造的。

通過(guò)給對(duì)象設(shè)置代理,我們可以攔截對(duì)象屬性的取值/賦值操作。

舉個(gè)例子:

const student = {
  age: 23
}
const handler = {
  get(target, prop) {
    console.log("讀值:", key, value);
    target[key] = value;
    return target[prop]
  },
  set(target, key, value) {
    console.log("設(shè)置值", key, value);
    target[key] = value;
    return true
  }
}
const proxy = new Proxy(studengt, handler)
console.log(proxy.age)//23
//proxy.age=32   //32

實(shí)現(xiàn)代碼

//Proxy+Reflect
let obj = {
  name: 'aaa',
  age: 18
}
//第一個(gè)參數(shù)為要代理的對(duì)象,第二個(gè)參數(shù)位hander
const proxy = new Proxy(obj, {
  //當(dāng)訪問(wèn)第一個(gè)屬性的時(shí)候會(huì)得到getter
  //同時(shí)會(huì)傳遞三個(gè)參數(shù)
  //target要進(jìn)行代理對(duì)象,這里就是obj
  //key被訪問(wèn)的屬性
  //receiver用來(lái)綁定this
  get(target, key, receiver) {
    console.log(`${key}屬性被訪問(wèn)`)
    return Reflect.get(target, key, receiver)
  },
  //當(dāng)某一屬性修改的時(shí),回來(lái)到Setter
  // 同時(shí)會(huì)傳遞四個(gè)參數(shù)
  // target要進(jìn)行代理的對(duì)象,這里就是obj
  // 可以被訪問(wèn)的屬性
  // newValue新修改的值
  // receiver用來(lái)綁定this
  set(target, key, newValue, receiver) {
    console.log(`${key}屬性修改`)
    return Reflect.set(target, key, newValue, receiver)
  }
})
// 以上代碼執(zhí)行完,得到的就是proxy對(duì)象就是obj對(duì)象的代理
// 我們只需要修改代理對(duì)象的就可以做到修改原型對(duì)象的效果
// 而且我們對(duì)代理對(duì)象的修改使我們能夠監(jiān)聽(tīng)到的
console.log(proxy.name)
proxy.name = 'bbb'

到此這篇關(guān)于詳解Vue響應(yīng)式的部分實(shí)現(xiàn)的文章就介紹到這了,更多相關(guān)Vue響應(yīng)式內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 從dubbo源碼分析qos-server端口沖突問(wèn)題及解決

    從dubbo源碼分析qos-server端口沖突問(wèn)題及解決

    這篇文章主要介紹了從dubbo源碼分析qos-server端口沖突問(wèn)題及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-02-02
  • Spring mvc Controller和RestFul原理解析

    Spring mvc Controller和RestFul原理解析

    這篇文章主要介紹了Spring mvc Controller和RestFul原理解析,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-03-03
  • Java接口方法默認(rèn)靜態(tài)實(shí)現(xiàn)代碼實(shí)例

    Java接口方法默認(rèn)靜態(tài)實(shí)現(xiàn)代碼實(shí)例

    這篇文章主要介紹了Java接口方法默認(rèn)靜態(tài)實(shí)現(xiàn),文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友可以參考下
    2020-06-06
  • Java中的MapStruct用法詳解

    Java中的MapStruct用法詳解

    這篇文章主要介紹了Java中的MapStruct用法詳解,MapStuct的使用非常簡(jiǎn)單,把對(duì)應(yīng)的jar包引入即可,本文通過(guò)示例代碼給大家介紹的非常詳細(xì),需要的朋友可以參考下
    2022-04-04
  • SpringBoot整合Aop全過(guò)程

    SpringBoot整合Aop全過(guò)程

    AOP(面向切面編程)技術(shù)可以高效地解決日志記錄、事務(wù)管理、權(quán)限控制等問(wèn)題,日志記錄通過(guò)自定義注解和切面類,自動(dòng)記錄方法調(diào)用詳情,減少重復(fù)代碼,事務(wù)管理方面,通過(guò)AOP可以在不改變業(yè)務(wù)代碼的情況下,實(shí)現(xiàn)事務(wù)的自動(dòng)開(kāi)啟、提交和回滾,保證數(shù)據(jù)一致性
    2024-10-10
  • SpringBoot整合Mybatis-plus案例及用法實(shí)例

    SpringBoot整合Mybatis-plus案例及用法實(shí)例

    mybatis-plus是一個(gè) Mybatis 的增強(qiáng)工具,在 Mybatis 的基礎(chǔ)上只做增強(qiáng)不做改變,為簡(jiǎn)化開(kāi)發(fā)、提高效率而生,下面這篇文章主要給大家介紹了關(guān)于SpringBoot整合Mybatis-plus案例及用法實(shí)例的相關(guān)資料,需要的朋友可以參考下
    2022-11-11
  • MyBatis一級(jí)與二級(jí)緩存相關(guān)配置

    MyBatis一級(jí)與二級(jí)緩存相關(guān)配置

    mybatis-plus是一個(gè)Mybatis的增強(qiáng)工具,在Mybatis的基礎(chǔ)上只做增強(qiáng)不做改變,為簡(jiǎn)化開(kāi)發(fā)、提高效率而生,這篇文章帶你了解Mybatis的一級(jí)和二級(jí)緩存
    2023-01-01
  • Spring-Data-JPA整合MySQL和配置的方法

    Spring-Data-JPA整合MySQL和配置的方法

    這篇文章主要介紹了Spring Data JPA整合MySQL和配置,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2018-04-04
  • SpringBoot使用AOP記錄接口操作日志的方法

    SpringBoot使用AOP記錄接口操作日志的方法

    日志記錄量是很大的,所以只記錄關(guān)鍵地方并按期歸檔,最好是存在如elasticsearch中,如果存在數(shù)據(jù)庫(kù)中,分表是不錯(cuò)的選擇,這篇文章主要介紹了SpringBoot使用AOP記錄接口操作日志的方法,需要的朋友可以參考下
    2022-08-08
  • 淺談Java的兩種多線程實(shí)現(xiàn)方式

    淺談Java的兩種多線程實(shí)現(xiàn)方式

    本篇文章主要介紹了淺談Java的兩種多線程實(shí)現(xiàn)方式,小編覺(jué)得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過(guò)來(lái)看看吧
    2017-08-08

最新評(píng)論