精品欧美一区二区三区在线观看 _久久久久国色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ù)棧公眾號

一区二区三区四区在线播放| 一区二区不卡| 欧美四级电影网| 在线视频不卡一区二区三区| 亚洲国产精品久久人人爱潘金莲| 国产人成精品一区二区三| 一区二区日韩精品| 国产女主播在线播放| 日韩天堂在线| 亚洲在线成人精品| 香蕉久久夜色| 欧美天堂在线视频| 男女男精品视频| 久久人人爽人人爽人人片av高清| 国产精品成人无码免费| 亚洲欧洲国产精品一区| 欧美午夜电影网| 亚洲精品久久久久久久蜜桃臀| 免费在线性爱视频| 国产成人啪午夜精品网站男同| 日韩美女毛茸茸| 久久99久久98精品免观看软件 | 日韩欧美国产综合一区 | 久久综合成人网| 色欧美自拍视频| 亚洲欧美在线播放| youjizz.com国产| 高清国产一区二区三区四区五区| 欧美性猛交xxxx| 国产精彩视频一区二区| www红色一片_亚洲成a人片在线观看_| 久久久久久亚洲综合影院红桃| 国产一区二区丝袜| 天堂av免费在线观看| 国模大胆一区二区三区| 精品国偷自产在线视频| 黄免费在线观看| 亚洲aa在线| 亚洲第一网中文字幕| 午夜大片在线观看| 久久久免费人体| 在线国产电影不卡| 国产精品97在线| 麻豆mv在线看| 午夜av一区二区三区| 999久久欧美人妻一区二区| 久做在线视频免费观看| 中文字幕av在线一区二区三区| 久久99影院| 十九岁完整版在线观看好看云免费| 丰满放荡岳乱妇91ww| 亚洲综合成人婷婷小说| 国产美女三级无套内谢| 国产一区在线视频| 91在线视频一区| 国产一区二区女内射| 麻豆极品一区二区三区| 国产在线拍揄自揄视频不卡99| 久久久精品毛片| 日本最新不卡在线| 国产欧美日韩专区发布| 曰批又黄又爽免费视频| 激情综合网最新| 成人激情在线播放| 精品人妻一区二区三区日产乱码 | 97久久夜色精品国产九色| 国产麻豆免费视频| 成人一区二区三区视频在线观看| 国产富婆一区二区三区 | 在线一区二区日韩| 97精品在线播放| 亚洲精品一区二区妖精| 色综合视频一区中文字幕| 麻豆疯狂做受xxxx高潮视频| 50路60路老熟妇啪啪| 丁香花五月婷婷| 精品国产一区二区三区香蕉沈先生| 精品视频在线观看日韩| 亚洲精品国产一区黑色丝袜| 日本不卡电影| 久久av.com| 国产成一区二区| 国产一级视频在线播放| 国产日韩一区二区三区在线播放| 清纯唯美日韩制服另类| 在线播放精品视频| 国产suv精品一区二区6| 久久久久久国产精品mv| 日本中文字幕在线观看| 一区二区三区小说| 国产日产欧美视频| 国产精品伊人| 亚洲电影在线看| 色综合99久久久无码国产精品| 91精品精品| 欧美又大又硬又粗bbbbb| 这里只有精品6| 丁香激情综合五月| 日本视频一区二区不卡| 91黄色在线| 91久久精品一区二区二区| 欧美国产在线一区| 欧美热在线视频精品999| 久久精品在线播放| 亚洲天堂av片| 国产美女一区二区| 欧美最大成人综合网| 亚洲按摩av| 欧美亚洲动漫另类| 国产乱淫av麻豆国产免费| 美女精品一区最新中文字幕一区二区三区| y97精品国产97久久久久久| 日本一二三区不卡| 国产一区二区中文字幕| 欧美日韩亚洲在线| 91视频欧美| 91精品国产福利在线观看| 日本激情小视频| 欧美不卡高清| 国产精品视频yy9099| 亚洲AV成人无码一二三区在线| 亚洲视频精选在线| 国产一级做a爰片久久| 久久精品论坛| 欧美精品中文字幕一区| 在线观看免费中文字幕| 久久久美女艺术照精彩视频福利播放| 97超碰在线视| 亚洲精品成a人ⅴ香蕉片| 亚洲欧美日韩天堂一区二区| 久久久精品99| 国产精品综合网| 一区二区精品在线| 精品欧美一区二区三区在线观看 | 成人免费在线观看入口| 激情网站五月天| 日韩在线黄色| 91av国产在线| 日韩欧美亚洲系列| 欧美日韩一区二区免费视频| 国产精品一区二区人妻喷水| 亚洲视频中文| 国产精品裸体一区二区三区| 午夜dj在线观看高清视频完整版| 91精品国产综合久久香蕉的特点| 国产91在线播放九色| 免费观看成人av| 台湾成人av| 欧美黄页在线免费观看| 中文字幕日韩精品在线| 中文字幕人妻互换av久久| 中文字幕av不卡| 天天干天天综合| 色999国产精品| 国产综合久久久久久| 日本a在线播放| 欧美一区二区三区播放老司机| 日韩一级片大全| 国产精品资源站在线| 天堂av在线中文| 凹凸av导航大全精品| 韩国v欧美v日本v亚洲| 香蕉视频网站在线| 色婷婷av一区二区三区之一色屋| 在线看片中文字幕| 国产一区二区三区精品视频| 屁屁影院ccyy国产第一页| 国产66精品| 国内精品一区二区三区| 视频二区在线| 欧美日韩国产小视频在线观看| 日韩一卡二卡在线观看| 国产激情91久久精品导航| 中国丰满熟妇xxxx性| 国产a久久精品一区二区三区| 国产精品久久激情| 国产欧美久久久久久久久| 欧美成人在线直播| 六月丁香激情综合| 国产精品毛片a∨一区二区三区| 久久久久久久久久久久久久久国产| 一本一本久久a久久综合精品| 成人91免费视频| 天堂网在线最新版www中文网| 亚洲性线免费观看视频成熟| 国产精品久久久久久在线| 亚洲第一激情av| 日本一级免费视频| 国产精品一区二区三区四区| 男人日女人逼逼| 水蜜桃久久夜色精品一区| 国产精华一区| 粉嫩av一区二区三区四区五区| 欧美精品在线看| 国产黄色免费在线观看| 精品三级av在线| 91青青草视频| 亚洲高清一区二区三区| 三区四区在线观看| 成人爱爱电影网址| 日本肉体xxxx裸体xxx免费| 亚洲承认在线| 久久av秘一区二区三区| 亚洲动漫精品| 成人三级在线| 啪啪av大全导航福利综合导航| 国内精品美女av在线播放| 999国产在线视频| 日韩成人中文电影| 精品人妻一区二区三区换脸明星| 在线免费一区三区| 亚洲男人第一av| 一区二区三区四区亚洲| 激情无码人妻又粗又大| 久久久久久久综合色一本| 亚洲精品激情视频| 国产在线播放一区三区四| 国产三级三级三级看三级| 999亚洲国产精| 激情五月六月婷婷| 亚洲国产精品91| 亚洲精品第一区二区三区| 欧美一区二区三区红桃小说| 产国精品偷在线| 日韩在线激情| 国产在线久久久| 久久久精品一区二区毛片免费看| 日韩av大片在线| 三级中文字幕在线观看| 欧美激情免费在线| 少女频道在线观看高清| 久久影院模特热| 日本高清在线观看wwwww色| 亚洲一区二区福利| 国自产拍在线网站网址视频| 日韩电影大全免费观看2023年上 | 亚洲美女又黄又爽在线观看| 欧美特黄一级视频| 亚洲国产成人精品久久久国产成人一区 | 香蕉久久网站| 杨幂一区欧美专区| 日韩一区自拍| 久久免费看毛片| 99热国内精品| 超碰免费在线公开| 99久久夜色精品国产亚洲1000部| 亚洲欧美久久234| 久久中文字幕av一区二区不卡| 亚洲一区二区四区| 久久综合国产| 宅男噜噜99国产精品观看免费| 久久中文字幕av| 日韩一二区视频| 欧美日韩综合| 国产黄页在线观看| 美女久久一区| 黄色在线视频网| 极品少妇一区二区三区精品视频| 三年中文在线观看免费大全中国| 精品一区二区在线视频| 污视频在线观看免费网站| 国产精品66部| 久久久国产精品无码| 91蝌蚪国产九色| 舐め犯し波多野结衣在线观看| 欧美国产精品久久| 久久中文免费视频| 午夜电影网亚洲视频| 视频一区二区三区四区五区| 91福利国产精品| 国产精品久久久午夜夜伦鲁鲁| 日韩亚洲欧美一区| 五月婷婷六月丁香| 伊人一区二区三区久久精品| 日韩在线观看www| 欧美高清视频在线观看| 韩国成人二区| 国产精品扒开腿做爽爽爽视频| 国产成人久久精品麻豆二区| 亚洲自拍av在线| 丝袜美腿一区二区三区动态图| 欧美日韩一区二区视频在线观看| 久久综合成人| 337p粉嫩大胆噜噜噜鲁| 九色porny丨国产精品| 成年人小视频在线观看| 国产区在线观看成人精品| 四虎884aa成人精品| 精品久久久久久亚洲精品| 中文字幕 日韩有码| 精品国产精品一区二区夜夜嗨| 久草在线免费福利资源| 欧美另类在线观看| av激情成人网| 成人片在线免费看| av一区二区在线播放| 日韩亚洲欧美视频| 秋霞午夜鲁丝一区二区老狼| 成人三级做爰av| 国产清纯白嫩初高生在线观看91 | ass极品国模人体欣赏| 一区二区三区丝袜| 婷婷激情五月综合| 精品国产一区二区三区四区四| 成a人v在线播放| 91超碰caoporn97人人| 精品视频一二| 日韩精品资源| 亚洲久久成人| 特种兵之深入敌后| 国产精品区一区二区三| 日本在线播放视频| 日韩美女在线视频| 免费av在线网址| 国产精品视频内| 窝窝社区一区二区| a天堂资源在线观看| 精品一区二区三区蜜桃| 在线观看免费小视频| 欧美午夜视频在线观看| 成人免费视频国产| 欧美成年人视频| 北岛玲精品视频在线观看| 五月天婷亚洲天综合网鲁鲁鲁| 日韩午夜av在线| 成人做爰www看视频软件| 亚洲激情图片小说视频| 在线免费观看中文字幕| 亚洲色图激情小说| 国产精品一区二区av影院萌芽| 久久精品人成| 在线视频日韩| 强迫凌虐淫辱の牝奴在线观看| 亚洲综合免费观看高清完整版在线| 91成人国产综合久久精品| 在线亚洲国产精品网| 精品日韩视频| 亚洲日本精品| 美国欧美日韩国产在线播放| 人妻视频一区二区| 在线看国产一区二区| av在线天堂播放| 国产精品老女人精品视频| 不卡一区综合视频| 日本黄色福利视频| 亚洲欧美在线另类| 精品国自产拍在线观看| 九九精品视频在线| 91精品导航| 国产3p露脸普通话对白| 99国产欧美久久久精品| 中文字幕av影院| 亚洲人午夜精品免费| 韩国成人在线| 中文字幕一区二区三区在线乱码| 久久99精品久久久久久久久久久久| 北条麻妃在线观看视频| 日韩一级片在线播放| 久操av在线| 久久涩涩网站| 蜜臀av一级做a爰片久久| fc2ppv在线播放| 精品久久久久香蕉网| 漫画在线观看av| 日韩在线第一区| 国产一区不卡在线| 免费黄色网址在线| 在线观看亚洲区| 欧美电影院免费观看| 国产婷婷一区二区三区| 久久久久久久久岛国免费| 亚洲系列第一页| 欧美激情一区二区三级高清视频| 精品精品国产毛片在线看| 久久国产色av免费观看| 中文字幕色av一区二区三区| 亚洲国产视频一区二区三区| 欧美性视频在线| 国产精品精品国产一区二区| 日韩高清一二三区| 欧美三级xxx| 欧美高清视频| 久久伊人一区二区| 精品一区二区在线看| 国产情侣在线视频| 日韩在线视频线视频免费网站| 操欧美女人视频| 欧美大尺度做爰床戏| 亚洲五码中文字幕| 自拍视频在线| 国产乱码精品一区二区三区卡| 日韩精品三区四区| 久久国产一级片| 在线看福利67194| 老司机精品视频在线播放| 色婷婷成人在线| 欧美日韩国产在线看| 国产成人午夜| 欧美一区二区三区在线播放| 国产精品一区二区三区网站|