精品欧美一区二区三区在线观看 _久久久久国色av免费观看性色_国产精品久久在线观看_亚洲第一综合网站_91精品又粗又猛又爽_小泽玛利亚一区二区免费_91亚洲精品国偷拍自产在线观看 _久久精品视频在线播放_美女精品久久久_欧美日韩国产成人在线

深入Vue3響應(yīng)式:手寫實現(xiàn)reactive與ref

開發(fā) 前端
本文將帶您深入實現(xiàn)其核心機制,我們將重點實現(xiàn)響應(yīng)式數(shù)據(jù)變化時的依賴收集與觸發(fā)更新功能,暫不涉及虛擬DOM和diff算法部分,視圖更新將直接使用DOM API實現(xiàn)。通過這個實現(xiàn),將更透徹地理解。

前言

上篇文章介紹了Vue3響應(yīng)式的兩個核心API,知道了兩者的用法于區(qū)別,本文將帶您深入實現(xiàn)其核心機制。我們將重點實現(xiàn)響應(yīng)式數(shù)據(jù)變化時的依賴收集與觸發(fā)更新功能,暫不涉及虛擬DOM和diff算法部分,視圖更新將直接使用DOM API實現(xiàn)。通過這個實現(xiàn),將更透徹地理解:

  1. 如何建立數(shù)據(jù)與視圖的響應(yīng)式關(guān)聯(lián)
  2. 依賴收集的核心原理
  3. 觸發(fā)更新的執(zhí)行機制

響應(yīng)式實現(xiàn)方案

關(guān)于響應(yīng)式方案,Vue目前一共出現(xiàn)過三種方案,分別是:

方案

版本

核心缺陷

defineProperty

Vue2

無法攔截數(shù)組操作、對象屬性增刪

Proxy + Reflect

Vue3

完美解決Vue2的響應(yīng)式限制

getter/setter

ref實現(xiàn)

支持基本數(shù)據(jù)類型的響應(yīng)式

defineProperty是Vue2中使用的響應(yīng)式方案,由于該API有挺多缺陷,Vue2底層對此做了許多處理,比如:

  • 對數(shù)組無法攔截
  • 對象屬性的新增與刪除無法攔截

所以Vue3選擇了使用Proxy這個核心API與對象的getter與setter,響應(yīng)式機制的主要功能就是,可以把普通的 JavaScript 對象封裝成為響應(yīng)式對象,攔截數(shù)據(jù)的獲取和修改操作,實現(xiàn)依賴數(shù)據(jù)的自動化更新。接下來我們嘗試動手實現(xiàn):

reactive

reactive是通過ES6中Proxy來實現(xiàn)屬性攔截的,所以我們可以先來實現(xiàn)一下:

const reactive =  <T extends object>(target: T) => {
    // 限制reactive只能傳遞引用類型,如果傳遞的不是引用類型,則出警告并將原始值直接返回
    if (typeof target !== 'object' || target === null) {
        console.warn('Reactive can only be applied to objects');
        return target
    }

    // 返回原始值的代理對象
    returnnew Proxy(target, {
        get(target, key, receiver) {
            const value = Reflect.get(target, key, receiver);
            // 這里需要收集依賴(后面實現(xiàn))
            track(target, key);
            // 如果值是對象,則遞歸調(diào)用reactive
            if (typeof value === 'object' && value !== null) {
                return reactive(value); 
            }
            
            return value;
        },
        set(target, key, value, receiver) {
            const result = Reflect.set(target, key, value, receiver);

            // 這里需要觸發(fā)更新(后面實現(xiàn))
            trigger(target, key)
            return result;
        },
    })
}

exportdefault reactive;

Proxy有許多攔截方法,這里我們暫時只需要攔截getset的操作

  • get方法中除了需要返回最新的數(shù)據(jù),還需要收集依賴
  • set方法中除了更新數(shù)據(jù),還需要執(zhí)行上面收集的依賴

核心架構(gòu):

圖片


track(依賴收集)

接著來實現(xiàn)一下track方法,該方法的主要作用就是收集依賴,這里可以使用Map去進行存儲依賴關(guān)系,Map的key就是我們的代理對象,而value還是一個嵌套的map,存儲代理對象的每個key以及對應(yīng)的依賴函數(shù)數(shù)組,因為每個key都可以有多個依賴

結(jié)構(gòu)如圖:

圖片


const targetMap = new WeakMap()
exportconst track = (target: object, key: PropertyKey) => {
    
    // 先找到target對應(yīng)的依賴
    let depsMap = targetMap.get(target)

    if(!depsMap) {
        // 如果沒找到,則說明是第一次收集,需要初始化
        depsMap = new Map()
        targetMap.set(target, depsMap)
    }
    // 接著需要對代理對象的屬性進行依賴收集
    let deps = depsMap.get(key)
    if(!deps) {
        deps = new Set()
    }
    if (!deps.has(activeEffect) && activeEffect) { 
        // 防止重復(fù)注冊 
        deps.add(activeEffect) 
        
    }
    depsMap.set(key, deps)
    console.log(`Tracking ${String(key)} on`, target);
};

trigger(更新觸發(fā))

實現(xiàn)完track方法后,我們再來實現(xiàn)一下trigger,該方法的主要作用就是從 targetMap 中,根據(jù) target 和 key 找到對應(yīng)的依賴函數(shù)集合 deps,然后遍歷 deps 執(zhí)行依賴函數(shù)

export const trigger = (target: object, key: PropertyKey) => {
    // 先找到target對應(yīng)的依賴map
    // console.log('----',targetMap)
    const depsMap = targetMap.get(target)
    if(!depsMap) return
    // 再找到對應(yīng)屬性的依賴
    const deps = depsMap.get(key)
    // 如果沒有依賴可執(zhí)行,則返回
    if(!deps) return
    // 最后遍歷整個依賴set分別執(zhí)行
    console.log('--deps', deps)
    deps.forEach(effect => {
        effect?.()
    })
};

effect(副作用管理)

最后我們再來實現(xiàn)effect副作用函數(shù),該副作用函數(shù)主要是在依賴更新的時候調(diào)用,它接受一個函數(shù),在被調(diào)用的時候執(zhí)行這個函數(shù)

在 effectFn 函數(shù)內(nèi)部,把函數(shù)賦值給全局變量 activeEffect;然后執(zhí)行 fn() 的時候,就會觸發(fā)響應(yīng)式對象的 get 函數(shù),get 函數(shù)內(nèi)部就會把 activeEffect 存儲到依賴地圖中,完成依賴的收集

let activeEffect
export const effect = (fn: () => void) => {
    const effectFn = () => {
        activeEffect = effectFn
        fn()
    }

    effectFn()
}

關(guān)鍵流程:當(dāng)effect執(zhí)行時,內(nèi)部函數(shù)會訪問響應(yīng)式數(shù)據(jù),觸發(fā)getter→track→將當(dāng)前effect存入依賴集合

驗證

響應(yīng)式底層的幾個核心方法都實現(xiàn)了,現(xiàn)在需要來驗證是否可行,比如:通過reactive處理的數(shù)據(jù),在數(shù)據(jù)更新時對應(yīng)頁面內(nèi)容也需要更新。

由于沒有寫虛擬DOM與diff算法的邏輯,所以更新的操作我們直接使用DOM API來代替,主要是驗證依賴收集與觸發(fā)更新的邏輯是否符合預(yù)期

<!DOCTYPE html>
<html lang="en">
<head>
    <meta charset="UTF-8">
    <meta name="viewport" content="width=device-width, initial-scale=1.0">
    <title>Document</title>
</head>
<body>
    <div id="app">
        <div id="content"></div>
        <button id="countBtn">count++</button>
    </div>
    <script type="module">
        // ts 部分先編譯成js
        import reactive from'./reactive/reactive.js';
        import { effect } from'./reactive/effect.js'
        // 通過自定義reactive創(chuàng)建響應(yīng)式數(shù)據(jù)
        const state = reactive({
            count: 0,
            name: '南玖'
        });
        // 注冊副作用函數(shù),更新視圖
        effect(() => {
            document.querySelector('#content').innerText = `name: ${state.name} --- car數(shù)量: ${state.count}`
        })
        // 按鈕點擊操作
        document.querySelector('#countBtn').addEventListener('click', () => {
            // 數(shù)據(jù)更新
            state.count += 1
        })
        console.log(state); // 0
    </script>
</body>
</html>


圖片


到這里reactive的響應(yīng)式原理就基本實現(xiàn)了,我們繼續(xù)來實現(xiàn)一下ref的響應(yīng)式邏輯

ref

相比reactiveref的實現(xiàn)原理更簡單一些,由于ref即可以傳遞基本數(shù)據(jù)類型也可以傳遞引用數(shù)據(jù)類型,而Proxy只能只能接受引用數(shù)據(jù)類型。所以ref采用的是面向?qū)ο蟮?getter 和 setter 攔截了 value 屬性的讀寫,這也是為什么我們 ref 數(shù)據(jù)的 需要通過.value訪問的原因

import { track, trigger } from'./effect'
import  reactive  from'./reactive'


const ref = (v) => {
    returnnew RefImpl(v)
}

class RefImpl {
    _value
    constructor(v) {
        this._value = convert(v)
    }

    get value() {
        track(this, 'value')
        returnthis._value
    }

    set value(val) {
        if(val === this._value) return
        this._value = convert(val)
        console.log('觸發(fā)更新')
        trigger(this, 'value')
    }
}


const convert = (v) => {
    return isObject(v) ? reactive(v) : v
}

const isObject = (v) => {
    returntypeof v === 'object' && v !== null
}

exportdefault ref


圖片


對于引用類型的數(shù)據(jù),ref底層會去調(diào)用reactive進行處理

總結(jié)

  1. 響應(yīng)式核心三角
    圖片
  2. reactive核心
  • 基于Proxy的深度代理
  • 嵌套對象自動響應(yīng)化
  • 使用WeakMap存儲依賴關(guān)系
  1. ref核心
  • getter/setter攔截value訪問

  • 基本類型與引用類型統(tǒng)一處理

  • 對象類型底層自動調(diào)用reactive

  1. 性能優(yōu)化點

  • 相同值不觸發(fā)更新

  • WeakMap避免內(nèi)存泄漏

  • 依賴函數(shù)精確收集

責(zé)任編輯:龐桂玉 來源: 前端南玖
相關(guān)推薦

2025-07-31 09:01:07

2022-06-26 00:00:02

Vue3響應(yīng)式系統(tǒng)

2022-12-06 08:39:27

Vue3Reactive

2021-12-02 05:50:35

Vue3 插件Vue應(yīng)用

2021-12-28 08:46:00

Vue3refreactive

2022-06-23 07:46:34

VueMobx系統(tǒng)

2024-04-11 13:10:00

Vue3Reactive響應(yīng)性

2020-06-09 11:35:30

Vue 3響應(yīng)式前端

2021-09-27 06:29:47

Vue3 響應(yīng)式原理Vue應(yīng)用

2025-08-06 13:39:39

Vue3React響應(yīng)性

2023-08-09 10:21:07

Vue 3Reactive

2022-01-19 18:05:47

Vue3前端代碼

2019-07-01 13:34:22

vue系統(tǒng)數(shù)據(jù)

2023-02-06 08:39:01

PreactVue3響應(yīng)式

2025-02-17 08:58:06

2023-12-06 07:43:56

Vue如何定義事件

2022-07-12 08:14:15

vue3refIsRef

2024-07-08 08:43:19

2023-06-13 08:01:27

qwikSolidJS?reactive

2024-05-06 00:00:00

RefReactive性能
點贊
收藏

51CTO技術(shù)棧公眾號

999精品视频在这里| 99re热视频| jizzjizzjizz欧美| 天天av天天翘天天综合网 | www.亚洲一区| 亚洲在线观看网站| 毛片在线网址| 久久久久久免费毛片精品| 国产精品99久久久久久久久久久久| 老司机精品免费视频| 91嫩草精品| 欧美亚洲禁片免费| 日本一区午夜艳熟免费| 国产黄色在线| 成人免费毛片片v| 国产精品毛片a∨一区二区三区|国| 成人一级黄色大片| 神马久久影院| 91精品国产综合久久久久久漫画 | 久久本道综合色狠狠五月| 最好看的日本字幕mv视频大全| 欧美在线二区| 一本大道亚洲视频| 污污污www精品国产网站| 天堂综合在线播放| 日韩欧美中文字幕在线观看 | 亚洲国产精品一区| 日韩午夜在线视频| 3d动漫精品啪啪一区二区下载| 国产在线不卡一区二区三区| 日本久久电影网| 人妻久久久一区二区三区| 成年人网站在线| 中文字幕免费在线观看视频一区| 国产一区二区精品免费| 国产ts变态重口人妖hd| 奇米色777欧美一区二区| 3344国产精品免费看| 欧美日韩在线视频免费| 91亚洲国产| 色系列之999| av电影网站在线观看| 日韩三级视频| 亚洲男人的天堂一区二区| 午夜影院日韩| 欧美一级片在线观看| mm1313亚洲国产精品无码试看| 国产乱码在线| 欧美aaaaa性bbbbb小妇| 综合精品久久| 在线观看久久久久久| 六十路息与子猛烈交尾| 日本高清精品| gogo久久| 粉嫩一区二区三区性色av| 成人性教育视频在线观看| 中文字幕第三页| 日韩中文字幕1| 久久午夜电影| 国产麻豆精品久久一二三| 欧美巨乳在线观看| 精品国产乱码久久久久久鸭王1 | 日韩av电影免费播放| 久久精品a一级国产免视看成人 | 97在线超碰| 黄色成人在线免费| 丰满人妻中伦妇伦精品app| 亚洲天堂手机| 欧美性生交xxxxxdddd| 国产最新免费视频| 亚洲va中文在线播放免费| 在线观看亚洲成人| 一女二男3p波多野结衣| 国产999精品在线观看| 日韩一区二区免费电影| 91porn在线| 天美av一区二区三区久久| 亚洲精品视频在线播放| 欧美激情 一区| 日韩精品久久| 欧美国产视频一区二区| 国产精彩视频在线观看| 校园激情久久| 成人av色在线观看| 欧洲av在线播放| 久久嫩草精品久久久精品| 神马影院午夜我不卡影院| 黄色免费在线观看| 午夜成人免费电影| 国产九九在线视频| jazzjazz国产精品麻豆| 一区二区三区黄色| 高h视频免费观看| 国产一区导航| 91色视频在线观看| 五月天婷婷在线观看| 国产欧美精品一区二区三区四区| 在线观看视频黄色| 这里有精品可以观看| 欧美日韩国产精品自在自线| 日韩女优在线视频| heyzo久久| 欧美高清在线播放| 中文字幕av影视| 成人动漫av在线| 五月天婷亚洲天综合网鲁鲁鲁| 在线中文字幕第一页| 欧美性少妇18aaaa视频| 337p日本欧洲亚洲大胆张筱雨 | 国产精品麻豆成人av电影艾秋| 精品日产卡一卡二卡麻豆| 永久免费毛片在线观看| 黄色成人在线网站| 成人免费看吃奶视频网站| 婷婷五月综合久久中文字幕| 亚洲欧洲日韩女同| av视屏在线播放| 国产美女撒尿一区二区| 久久精品国产综合| 免费观看日批视频| bt欧美亚洲午夜电影天堂| 中文字幕久久一区| 日韩美女在线看免费观看| 精品成人免费观看| caoporn91| 另类中文字幕网| 欧洲av一区| 九色porny视频在线观看| 日韩欧美在线综合网| 欧美做受高潮6| 亚洲一区二区网站| 国产精品加勒比| 丝袜在线视频| 欧美一区二区视频在线观看 | 一区二区三区久久| 永久免费的av网站| 色婷婷综合网| 国产精品极品美女在线观看免费| 亚洲日本中文字幕在线| 五月天视频一区| 亚洲精品国产成人av在线| 欧美日韩mv| 97se亚洲综合| 精品国产99久久久久久| 欧美日韩国产免费一区二区 | 天天色棕合合合合合合合| 亚洲午夜羞羞片| 无码成人精品区在线观看| 狠狠久久婷婷| 国产在线欧美日韩| 蜜桃av在线播放| 亚洲老头同性xxxxx| 欧美日韩乱国产| 91视频免费播放| 97成人在线观看视频| 网友自拍区视频精品| 51精品国产黑色丝袜高跟鞋| 瑟瑟在线观看| 日本高清无吗v一区| 国产毛片欧美毛片久久久| 久久精品国产久精国产| www亚洲国产| 嫩呦国产一区二区三区av| 欧美成在线视频| 国产1区在线观看| 婷婷久久综合九色综合伊人色| 少妇饥渴放荡91麻豆| 久久亚洲美女| 午夜精品亚洲一区二区三区嫩草| 久久精品xxxxx| 精品久久久av| 男人天堂手机在线观看| 欧美日韩免费网站| 性欧美一区二区| 激情欧美一区二区三区在线观看| 欧美一区二区三区综合| 国语一区二区三区| 国产精品jizz在线观看麻豆| 免费大片黄在线观看视频网站| 日韩欧美成人一区二区| 国产成人无码精品亚洲| 国产欧美一二三区| 女王人厕视频2ⅴk| 性伦欧美刺激片在线观看| 午夜一区二区三区| 亚洲视频国产精品| 欧美壮男野外gaytube| 美女写真理伦片在线看| 亚洲国产精品电影| 中文字幕 日韩有码| 亚洲一区日韩精品中文字幕| 88久久精品无码一区二区毛片| 麻豆成人久久精品二区三区红| 日韩免费在线观看av| 最新国产一区| 97超碰最新| 极品美女一区| 欧美贵妇videos办公室| 成人在线免费公开观看视频| 欧美r级电影在线观看| 蜜臀99久久精品久久久久小说| 亚洲精品成人少妇| 免费看黄色av| proumb性欧美在线观看| 三级性生活视频| 美女爽到呻吟久久久久| 日本黄大片在线观看| 日本女优一区| 极品日韩久久| 涩爱av色老久久精品偷偷鲁| 国产精品第一视频| 国产高潮在线| 欧美寡妇偷汉性猛交| 美女羞羞视频在线观看| 伊人久久免费视频| 香蕉久久一区二区三区| 日韩视频免费观看高清完整版在线观看 | www.五月婷| 欧美日韩国产综合一区二区 | 韩国理伦片一区二区三区在线播放 | 99久久精品免费看国产四区 | 怡红院红怡院欧美aⅴ怡春院| 亚洲欧美三级伦理| 涩涩视频免费看| 91精品国产色综合久久ai换脸 | 日本vs亚洲vs韩国一区三区二区| 加勒比成人在线| 欧美aⅴ99久久黑人专区| 亚洲精品成人久久久998| 午夜a一级毛片亚洲欧洲| 国产高清一区二区三区| 精品一区二区三区视频在线播放 | 波多野结衣乳巨码无在线| 中文字幕一区二区三区在线视频| 亚洲欧美国产不卡| 欧美一区二区三区高清视频| 欧美一区二区视频17c| 欧美激情99| 精品国产乱码久久久久| 久久久亚洲欧洲日产| 国产精品一码二码三码在线| 在线综合色站| 国产精品一区二区欧美黑人喷潮水| 视频亚洲一区二区| 97视频资源在线观看| 日韩精品成人| 成人欧美一区二区| 国产精品乱战久久久| 国产精品一区视频| 欧美亚洲tv| 蜜桃成人免费视频| 九色精品国产蝌蚪| 日本欧美色综合网站免费| 国产一区99| 亚洲精品电影在线一区| 色爱综合网欧美| 国产盗摄视频在线观看| 欧美午夜一区| 色综合久久久久无码专区| 久久伊人亚洲| 日韩精品你懂的| 久久99精品久久久久婷婷| 一级 黄 色 片一| 国产成人综合亚洲91猫咪| 性感美女一区二区三区| 99精品在线免费| 性欧美精品男男| 日韩毛片视频在线看| 精品无码久久久久久久| 精品福利免费观看| 五月婷婷丁香在线| 在线成人av网站| 好吊视频一二三区| 亚洲欧洲第一视频| 免费在线观看黄| 国模精品系列视频| 午夜av成人| 亚洲a级在线播放观看| 国产精品15p| 欧美一区免费视频| 一本一本久久a久久综合精品| 欧美亚洲色图视频| 日韩成人免费看| 一区二区在线免费观看视频| 91亚洲国产成人精品一区二三 | 91精品久久久久久久| gogo人体一区| 亚洲一区精彩视频| 亚洲精品影视| 高清av免费看| 不卡在线视频中文字幕| 日本高清黄色片| 亚洲国产日日夜夜| 中文字幕+乱码+中文| 亚洲高清色综合| 黄色网页在线播放| 国产suv精品一区二区三区88区| 久久久国产精品入口麻豆| 精品欧美国产| 亚洲网色网站| 国产情侣av自拍| 成人成人成人在线视频| 国产精品久久久免费看| 日韩欧美成人精品| 国产免费视频一区二区三区| 亚洲欧美三级伦理| 91桃色在线观看| 成人夜晚看av| 精品久久一区| 欧美视频在线播放一区| 国产一区二区免费看| 制服 丝袜 综合 日韩 欧美| 亚洲成人资源在线| 国产免费黄色片| 中文字幕一精品亚洲无线一区 | 丝袜美腿美女被狂躁在线观看| 国产综合在线看| 精品一区二区三区中文字幕| 亚洲成人第一| 久久综合伊人| 影音先锋黄色资源| 亚洲午夜激情av| 精品国产九九九| 久久精品视频免费播放| 97人人做人人爽香蕉精品| 欧美精品久久久| 99国产精品自拍| www.555国产精品免费| 亚洲另类一区二区| 国产精品特级毛片一区二区三区| 国产亚洲欧美视频| 男人最爱成人网| 欧美日韩精品免费观看| 国产精品日韩久久久| 影音先锋黄色资源| 图片区小说区区亚洲影院| 亚洲黄色在线播放| 欧美大成色www永久网站婷| 在线不卡一区| 中文字幕日韩精品久久| 久久精品国产99国产精品| 亚洲精品成人av久久| 在线视频一区二区三区| 国产在线网站| 国产97在线亚洲| av伊人久久| 中文字幕资源在线观看| 《视频一区视频二区| 国产熟女一区二区丰满| 欧美刺激性大交免费视频| 亚洲视频一起| 隔壁人妻偷人bd中字| 成人h动漫精品一区二| 97人人澡人人爽人人模亚洲| 亚洲精品一区二区在线| av激情成人网| 亚洲啪啪av| 国产自产v一区二区三区c| 精品欧美一区二区久久久久| 日韩精品中午字幕| cao在线视频| 欧美极品日韩| 秋霞午夜av一区二区三区| 欧美肥妇bbwbbw| 精品少妇一区二区三区日产乱码| brazzers在线观看| 免费h精品视频在线播放| 日韩av午夜在线观看| 青青操在线视频观看| 欧美一级二级在线观看| 91色在线看| 日韩一区不卡| 国产精品自拍网站| 亚洲精品国产精品乱码| 国产一区二区三区18| 国产欧美视频在线| 亚洲中文字幕无码av永久| 国产欧美精品一区二区色综合 | 欧美成人高潮一二区在线看| 久久久影视传媒| 国产婷婷一区二区三区久久| 国模极品一区二区三区| 日韩av久操| 久久久久久久穴| 欧美性欧美巨大黑白大战| 超碰超碰在线| 麻豆亚洲一区| 国产一区二区不卡在线| 久久99国产综合精品免费| 久久精品国产99国产精品澳门| 成人av婷婷| 孩娇小videos精品| 亚洲福利视频导航| 8888四色奇米在线观看| 国产精品9999久久久久仙踪林 | 91亚洲精品在线观看| 国产精品日韩精品欧美精品| 侵犯稚嫩小箩莉h文系列小说| 日韩va亚洲va欧洲va国产| 亚洲伊人精品酒店|