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

vue3.0響應(yīng)式函數(shù)原理詳細(xì)

 更新時(shí)間:2022年02月14日 11:27:45   作者:隱冬  
這篇文章主要介紹了vue3.0響應(yīng)式函數(shù)原理,Vue3的響應(yīng)式系統(tǒng)可以監(jiān)聽(tīng)動(dòng)態(tài)添加的屬性還可以監(jiān)聽(tīng)屬性的刪除操作,以及數(shù)組的索引以及l(fā)ength屬性的修改操作。另外Vue3的響應(yīng)式系統(tǒng)還可以作為模塊單獨(dú)使用。下面更多介紹,需要的小伙伴可以才可以參考一下

前言:

Vue3重寫(xiě)了響應(yīng)式系統(tǒng),和Vue2相比底層采用Proxy對(duì)象實(shí)現(xiàn),在初始化的時(shí)候不需要遍歷所有的屬性再把屬性通過(guò)defineProperty轉(zhuǎn)換成get和set。另外如果有多層屬性嵌套的話只有訪問(wèn)某個(gè)屬性的時(shí)候才會(huì)遞歸處理下一級(jí)的屬性所以Vue3中響應(yīng)式系統(tǒng)的性能要比Vue2好。

接下來(lái)我們自己實(shí)現(xiàn)Vue3響應(yīng)式系統(tǒng)的核心函數(shù)(reactive/ref/toRefs/computed/effect/track/trigger)來(lái)學(xué)習(xí)一下響應(yīng)式原理。

首先我們使用Proxy來(lái)實(shí)現(xiàn)響應(yīng)式中的第一個(gè)函數(shù)reactive

1.reactive

reactive接收一個(gè)參數(shù),首先要判斷這個(gè)參數(shù)是否是一個(gè)對(duì)象,如果不是直接返回,reactive只能將對(duì)象轉(zhuǎn)換成響應(yīng)式對(duì)象,這是和ref不同的地方。

接著會(huì)創(chuàng)建攔截器對(duì)象handler, 其中抱哈get,set,deleteProperty等攔截方法,最后創(chuàng)建并返回Proxy對(duì)象。

// 判斷是否是一個(gè)對(duì)象
const isObject = val => val !== null && typeof val === 'object'
// 如果是對(duì)象則調(diào)用reactive
const convert= target => isObject(target) ? reactive(target) : target
// 判斷對(duì)象是否存在key屬性
const haOwnProperty = Object.prototype.hasOwnProperty
const hasOwn = (target, key) => haOwnProperty.call(target, key)

export function reactive (target) {
? ? if (!isObject(target)) {
? ? ? ? // 如果不是對(duì)象直接返回
? ? ? ? return target
? ? }

? ? const handler = {
? ? ? ? get (target, key, receiver) {
? ? ? ? ? ? // 收集依賴(lài)
? ? ? ? ? ? const result = Reflect.get(target, key, receiver)
? ? ? ? ? ? // 如果屬性是對(duì)象則需要遞歸處理
? ? ? ? ? ? return convert(result)
? ? ? ? },
? ? ? ? set (target, key, value, receiver) {
? ? ? ? ? ? const oldValue = Reflect.get(target, key, receiver)
? ? ? ? ? ? let result = true;
? ? ? ? ? ? // 需要判斷當(dāng)前傳入的新值和oldValue是否相等,如果不相等再去覆蓋舊值,并且觸發(fā)更新
? ? ? ? ? ? if (oldValue !== value) {
? ? ? ? ? ? ? ? result = Reflect.set(target, key, value, receiver)
? ? ? ? ? ? ? ? // 觸發(fā)更新...
? ? ? ? ? ? }
? ? ? ? ? ? // set方法需要返回布爾值
? ? ? ? ? ? return result;
? ? ? ? },
? ? ? ? deleteProperty (target, key) {
? ? ? ? ? ? // 首先要判斷當(dāng)前target中是否有自己的key屬性
? ? ? ? ? ? // 如果存在key屬性,并且刪除要觸發(fā)更新
? ? ? ? ? ? const hasKey = hasOwn(target, key)
? ? ? ? ? ? const result = Reflect.deleteProperty(target, key)
? ? ? ? ? ? if (hasKey && result) {
? ? ? ? ? ? ? ? // 觸發(fā)更新...
? ? ? ? ? ? }
? ? ? ? ? ? return result;
? ? ? ? }
? ? }
? ? return new Proxy(target, handler)
}

至此reactive函數(shù)就寫(xiě)完了,接著我們來(lái)編寫(xiě)一下收集依賴(lài)的過(guò)程。

在依賴(lài)收集的過(guò)程會(huì)創(chuàng)建三個(gè)集合,分別是targetMap,depsMap以及dep。

其中targetMap是用來(lái)記錄目標(biāo)對(duì)象和字典他使用的是weakMap,key是目標(biāo)對(duì)象,targetMap的值是depsMap, 類(lèi)型是Map,這里面的key是目標(biāo)對(duì)象的屬性名稱(chēng),值是一個(gè)Set集合,集合中存儲(chǔ)的元素是Effect函數(shù)。因?yàn)榭梢远啻握{(diào)用同一個(gè)Effect在Effect訪問(wèn)同一個(gè)屬性,這個(gè)時(shí)候這個(gè)屬性會(huì)收集多次依賴(lài)對(duì)應(yīng)多個(gè)Effect函數(shù)。

一個(gè)屬性可以對(duì)應(yīng)多個(gè)Effect函數(shù),觸發(fā)更新的時(shí)候可以通過(guò)屬性找到對(duì)應(yīng)的Effect函數(shù),進(jìn)行執(zhí)行。

我們這里分別來(lái)實(shí)現(xiàn)一下effecttrack兩個(gè)函數(shù)。

effect函數(shù)接收一個(gè)函數(shù)作為參數(shù),我們首先在外面定一個(gè)變量存儲(chǔ)callback, 這樣track函數(shù)就可以訪問(wèn)到callback了。

let activeEffect = null;
export function effect (callback) {
? ? activeEffect = callback;
? ? // 訪問(wèn)響應(yīng)式對(duì)象屬性,收集依賴(lài)
? ? callback();
? ? // 依賴(lài)收集結(jié)束要置null
? ? activeEffect = null;
}

track函數(shù)接收兩個(gè)參數(shù)目標(biāo)對(duì)象和屬性, 他的內(nèi)部要將target存儲(chǔ)到targetMap中。需要先定義一個(gè)這樣的Map。

let targetMap = new WeakMap()

export function track (target, key) {
? ? // 判斷activeEffect是否存在
? ? if (!activeEffect) {
? ? ? ? return;
? ? }
? ? // depsMap存儲(chǔ)對(duì)象和effect的對(duì)應(yīng)關(guān)系
? ? let depsMap = targetMap.get(target)
? ? // 如果不存在則創(chuàng)建一個(gè)map存儲(chǔ)到targetMap中
? ? if (!depsMap) {
? ? ? ? targetMap.set(target, (depsMap = new Map()))
? ? }
? ? // 根據(jù)屬性查找對(duì)應(yīng)的dep對(duì)象
? ? let dep = depsMap.get(key)
? ? // dep是一個(gè)集合,用于存儲(chǔ)屬性所對(duì)應(yīng)的effect函數(shù)
? ? if (!dep) {
? ? ? ? // 如果不存在,則創(chuàng)建一個(gè)新的集合添加到depsMap中
? ? ? ? depsMap.set(key, (dep = new Set()))
? ? }
? ? dep.add(activeEffect)
}

track是依賴(lài)收集的函數(shù)。需要在reactive函數(shù)的get方法中調(diào)用。

get (target, key, receiver) {
? ? // 收集依賴(lài)
? ? track(target, key)
? ? const result = Reflect.get(target, key, receiver)
? ? // 如果屬性是對(duì)象則需要遞歸處理
? ? return convert(result)
},

這樣整個(gè)依賴(lài)收集就完成了。接著就要實(shí)現(xiàn)觸發(fā)更新,對(duì)應(yīng)的函數(shù)是trigger,這個(gè)過(guò)程和track的過(guò)程正好相反。

trigger函數(shù)接收兩個(gè)參數(shù),分別是target和key。

export function trigger (target, key) {
? ? const depsMap = targetMap.get(target)
? ? // 如果沒(méi)有找到直接返回
? ? if (!depsMap) {
? ? ? ? return;
? ? }
? ? const dep = depsMap.get(key)
? ? if (dep) {
? ? ? ? dep.forEach(effect => {
? ? ? ? ? ? effect()
? ? ? ? })
? ? }
}

trigger函數(shù)要在reactive函數(shù)中的setdeleteProperty中觸發(fā)。

set (target, key, value, receiver) {
? ? const oldValue = Reflect.get(target, key, receiver)
? ? let result = true;
? ? // 需要判斷當(dāng)前傳入的新值和oldValue是否相等,如果不相等再去覆蓋舊值,并且觸發(fā)更新
? ? if (oldValue !== value) {
? ? ? ? result = Reflect.set(target, key, value, receiver)
? ? ? ? // 觸發(fā)更新...
? ? ? ? trigger(target, key)
? ? }
? ? // set方法需要返回布爾值
? ? return result;
},
deleteProperty (target, key) {
? ? // 首先要判斷當(dāng)前target中是否有自己的key屬性
? ? // 如果存在key屬性,并且刪除要觸發(fā)更新
? ? const hasKey = hasOwn(target, key)
? ? const result = Reflect.deleteProperty(target, key)
? ? if (hasKey && result) {
? ? ? ? // 觸發(fā)更新...
? ? ? ? trigger(target, key)
? ? }
? ? return result;
}

2.ref

ref接收一個(gè)參數(shù)可以是原始值也可以是一個(gè)對(duì)象,如果傳入的是對(duì)象并且是ref創(chuàng)建的對(duì)象則直接返回,如果是普通對(duì)象則調(diào)用reactive來(lái)創(chuàng)建響應(yīng)式對(duì)象,否則創(chuàng)建一個(gè)只有value屬性的響應(yīng)式對(duì)象。

export function ref (raw) {
? ? // 判斷raw是否是ref創(chuàng)建的對(duì)象,如果是直接返回
? ? if (isObject(raw) && raw.__v__isRef) {
? ? ? ? return raw
? ? }

? ? // 之前已經(jīng)定義過(guò)convert函數(shù),如果參數(shù)是對(duì)象就會(huì)調(diào)用reactive函數(shù)創(chuàng)建響應(yīng)式
? ? let value = convert(raw);

? ? const r = {
? ? ? ? __v__isRef: true,
? ? ? ? get value () {
? ? ? ? ? ? track(r, 'value')
? ? ? ? ? ? return value
? ? ? ? },
? ? ? ? set value (newValue) {
? ? ? ? ? ? // 判斷新值和舊值是否相等
? ? ? ? ? ? if (newValue !== value) {
? ? ? ? ? ? ? ? raw = newValue
? ? ? ? ? ? ? ? value = convert(raw)
? ? ? ? ? ? ? ? // 觸發(fā)更新
? ? ? ? ? ? ? ? trigger(r, 'value')
? ? ? ? ? ? }
? ? ? ? }
? ? }

? ? return r
}

3.toRefs

toRefs接收reactive函數(shù)返回的響應(yīng)式對(duì)象,如果不是響應(yīng)式對(duì)象則直接返回。將傳入對(duì)象的所有屬性轉(zhuǎn)換成一個(gè)類(lèi)似ref返回的對(duì)象將準(zhǔn)換后的屬性掛載到一個(gè)新的對(duì)象上返回。

export function toRefs (proxy) {
? ? // 如果是數(shù)組創(chuàng)建一個(gè)相同長(zhǎng)度的數(shù)組,否則返回一個(gè)空對(duì)象
? ? const ret = proxy instanceof Array ? new Array(proxy.length) : {}

? ? for (const key in proxy) {
? ? ? ? ret[key] = toProxyRef(proxy, key)
? ? }

? ? return ret;
}

function toProxyRef (proxy, key) {
? ? const r = {
? ? ? ? __v__isRef: true,
? ? ? ? get value () { // 這里已經(jīng)是響應(yīng)式對(duì)象了,所以不需要再收集依賴(lài)了
? ? ? ? ? ? return proxy[key]
? ? ? ? },
? ? ? ? set value (newValue) {
? ? ? ? ? ? proxy[key] = newValue
? ? ? ? }
? ? }
? ? return r
}

toRefs的作用其實(shí)是將reactive中的每個(gè)屬性都變成響應(yīng)式的。reactive方法會(huì)創(chuàng)建一個(gè)響應(yīng)式的對(duì)象,但是如果將reactive返回的對(duì)象進(jìn)行解構(gòu)使用就不再

是響應(yīng)式了,toRefs的作用就是支持解構(gòu)之后仍舊為響應(yīng)式。

4.computed

接著再來(lái)模擬一下computed函數(shù)的內(nèi)部實(shí)現(xiàn)

computed需要接收一個(gè)有返回值的函數(shù)作為參數(shù),這個(gè)函數(shù)的返回值就是計(jì)算屬性的值,需要監(jiān)聽(tīng)函數(shù)內(nèi)部響應(yīng)式數(shù)據(jù)的變化,最后將函數(shù)執(zhí)行的結(jié)果返回。

export function computed (getter) {
? ? const result = ref()

? ? effect(() => (result.value = getter()))

? ? return result
}

computed函數(shù)會(huì)通過(guò)effect監(jiān)聽(tīng)getter內(nèi)部響應(yīng)式數(shù)據(jù)的變化,因?yàn)樵趀ffect中執(zhí)行g(shù)etter的時(shí)候訪問(wèn)響應(yīng)式數(shù)據(jù)的屬性會(huì)去收集依賴(lài),當(dāng)數(shù)據(jù)變化會(huì)重新執(zhí)行effect函數(shù),將getter的結(jié)果再存儲(chǔ)到result中。

到此這篇關(guān)于vue3.0響應(yīng)式函數(shù)原理詳細(xì)的文章就介紹到這了,更多相關(guān)vue3.0響應(yīng)式函數(shù)原理內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • VUE3中的函數(shù)的聲明和使用

    VUE3中的函數(shù)的聲明和使用

    這篇文章主要介紹了VUE3中的函數(shù)的聲明和使用方式,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-10-10
  • Vue.extend構(gòu)造器的詳解

    Vue.extend構(gòu)造器的詳解

    這篇文章主要介紹了Vue.extend構(gòu)造器的詳解的相關(guān)資料,需要的朋友可以參考下
    2017-07-07
  • 詳解Vue組件如何正確引用和使用外部方法

    詳解Vue組件如何正確引用和使用外部方法

    在開(kāi)發(fā)Vue應(yīng)用時(shí),我們經(jīng)常需要在多個(gè)組件中復(fù)用一些通用的函數(shù)或方法,這些函數(shù)可能是我們自己編寫(xiě)的工具函數(shù),也可能是從第三方庫(kù)中導(dǎo)入的,下面我們就來(lái)看看如何正確引用和使用外部方法吧
    2024-01-01
  • 一文帶你詳細(xì)了解vue axios的封裝

    一文帶你詳細(xì)了解vue axios的封裝

    對(duì)請(qǐng)求的封裝在實(shí)際項(xiàng)目中是十分必要的,它可以讓我們統(tǒng)一處理 http 請(qǐng)求,比如做一些攔截,處理一些錯(cuò)誤等,本篇文章將詳細(xì)介紹如何封裝 axios 請(qǐng)求,需要的朋友可以參考下
    2023-09-09
  • vue?button的@click方法無(wú)效鉤子函數(shù)沒(méi)有執(zhí)行問(wèn)題

    vue?button的@click方法無(wú)效鉤子函數(shù)沒(méi)有執(zhí)行問(wèn)題

    這篇文章主要介紹了vue?button的@click方法無(wú)效鉤子函數(shù)沒(méi)有執(zhí)行問(wèn)題及解決,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-03-03
  • vue如何在data中引入圖片的正確路徑

    vue如何在data中引入圖片的正確路徑

    這篇文章主要介紹了vue如何在data中引入圖片的正確路徑,具有很好的參考價(jià)值,希望對(duì)大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-06-06
  • 解決threeJS加載obj?gltf模型后顏色太暗的方法

    解決threeJS加載obj?gltf模型后顏色太暗的方法

    這篇文章主要為大家介紹了解決threeJS加載obj?gltf模型后顏色太暗的方法示例,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪
    2022-07-07
  • el-select綁定值遇到的問(wèn)題小結(jié)

    el-select綁定值遇到的問(wèn)題小結(jié)

    碰到一個(gè)問(wèn)題,選擇框的數(shù)據(jù)是后端傳過(guò)來(lái)的,下拉框的數(shù)據(jù)也是后端傳過(guò)來(lái)的,但是打開(kāi)下拉框時(shí),發(fā)現(xiàn)數(shù)據(jù)沒(méi)有高亮,最后通過(guò)只要選擇框v-model給的值和option的value綁定的值一致,就可以高亮,感興趣的朋友一起看看吧
    2023-12-12
  • Vue3框架使用報(bào)錯(cuò)以及解決方案

    Vue3框架使用報(bào)錯(cuò)以及解決方案

    這篇文章主要介紹了Vue3框架使用報(bào)錯(cuò)以及解決方案,具有很好的參考價(jià)值,希望對(duì)大家有所幫助,如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2023-10-10
  • Vue3中watch無(wú)法監(jiān)聽(tīng)的解決辦法

    Vue3中watch無(wú)法監(jiān)聽(tīng)的解決辦法

    本文主要介紹了Vue3中watch無(wú)法監(jiān)聽(tīng)的解決辦法,文中通過(guò)示例代碼介紹的非常詳細(xì),對(duì)大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價(jià)值,需要的朋友們下面隨著小編來(lái)一起學(xué)習(xí)學(xué)習(xí)吧
    2023-05-05

最新評(píng)論