真实的国产乱ⅩXXX66竹夫人,五月香六月婷婷激情综合,亚洲日本VA一区二区三区,亚洲精品一区二区三区麻豆

成都創(chuàng)新互聯(lián)網(wǎng)站制作重慶分公司

vue3原始值響應(yīng)丟失如何解決

今天小編給大家分享一下vue3原始值響應(yīng)丟失如何解決的相關(guān)知識點,內(nèi)容詳細(xì),邏輯清晰,相信大部分人都還太了解這方面的知識,所以分享這篇文章給大家參考一下,希望大家閱讀完這篇文章后有所收獲,下面我們一起來了解一下吧。

成都創(chuàng)新互聯(lián)公司主要從事網(wǎng)站設(shè)計、成都網(wǎng)站建設(shè)、網(wǎng)頁設(shè)計、企業(yè)做網(wǎng)站、公司建網(wǎng)站等業(yè)務(wù)。立足成都服務(wù)贛縣,十載網(wǎng)站建設(shè)經(jīng)驗,價格優(yōu)惠、服務(wù)專業(yè),歡迎來電咨詢建站服務(wù):18982081108

    一、ref 的引入

    ref 就是解決 proxy 無法直接代理原始值的問題。我們先來看 ref 的使用:

    const name = ref('小黑子')

    ref 是怎么實現(xiàn)的呢?其實就是用對象“包裹”原始值。我們再來看一下 ref 的實現(xiàn):

    function ref(val){
        // 使用對象包裹原始值
        const wrapper = {
            value:val    
        }
        // 利用 reactive 將對象變成響應(yīng)式數(shù)據(jù)
        return reactive(wrapper)
    }

    ref 的實現(xiàn)就是這么簡單。

    ref 對原始值響應(yīng)主要就做了這兩件事:

    • 1、使用對象包裹原始值。

    • 2、使用 reactive 將包裹對象變?yōu)轫憫?yīng)式數(shù)據(jù)。

    二、isref 的實現(xiàn)

    我們使用 ref 創(chuàng)建一個響應(yīng)式對象,但是我們要怎么區(qū)別一個對象是普通對象還是 ref 對象呢?于是我們的 isref 出現(xiàn)了。

    我們看一下它的使用:

    const name = ref('cj') console.log(isRef(name)); // true

    那么它的實現(xiàn)原理是怎樣的呢?主要實現(xiàn)還是在 ref API內(nèi)部,我們來看一下具體實現(xiàn)代碼:

    function ref(val){
        const wrapper = {
            value:val    
        }
        Object.defineProperty(warpper,'__v_isRef',{
            value:true    
        })
        return reactive(wrapper)
    }

    原來就是在 ref 內(nèi)部給 包裹對象添加一個不可枚舉不可寫的屬性,并且值為 true 。這樣我們就可以檢查該屬性來判斷是不是 ref 了。

    function isRef(val) {
        return val.__v_isRef ?? false
    }

    三、響應(yīng)丟失

    什么是響應(yīng)丟失?響應(yīng)丟失就是響應(yīng)式數(shù)據(jù)不進(jìn)行響應(yīng)了。我們來看下方代碼:

    const obj = reactive({foo:1,bar:2})
    const {foo,bar} = obj
    obj.foo++    // foo不會改變,還是 1

    上面的 obj 已經(jīng)響應(yīng)丟失了,也就不會觸發(fā)重新渲染。為什么會這樣呢?其實就是因為使用了結(jié)構(gòu)賦值,展開運算符也會使其失效。

    const obj = reactive({foo:1,bar:2})
    const newObj = {...obj}
    obj.foo++   // newObj.foo不會改變,還是 1

    這就相當(dāng)于重新定義了新的數(shù)據(jù),而不再是原來的響應(yīng)數(shù)據(jù)了,自然也就不具有響應(yīng)式能力。

    1、toRef登場

    toRef 就是為了解決響應(yīng)丟失的問題。我們來看一下它的實現(xiàn):

    function toRef(obj,key) {
        const wrapper = {
            get value() {
                return obj[key]        
            },
            set value(val) {
                obj[key] = val
            }    
        }
        Object.defineProperty(wrapper,'__v_isRef',{
            value:true    
        })
        return wrapper
    }

    傳入兩個參數(shù),第一個是響應(yīng)式數(shù)據(jù),第二個是 obj 的一個鍵。

    • 第一部分就是設(shè)置聲明一個對象,對象先設(shè)置了 value 屬性的 get 用于將訪問 toRef 值時,將返回傳入的響應(yīng)式數(shù)據(jù)對應(yīng)的屬性值,然后設(shè)置了 value屬性的 set 用于將設(shè)置 toRef 值時,就拿取設(shè)置的新值更新響應(yīng)式數(shù)據(jù)對應(yīng)的屬性值。也就是說,toRef 返回的對象還是利用的響應(yīng)式數(shù)據(jù)。

    • 第二部分用于設(shè)置返回的數(shù)據(jù)是 ref 數(shù)據(jù)。因為 toRef 返回的數(shù)據(jù)類似 ref 數(shù)據(jù),為了統(tǒng)一就直接認(rèn)定為是一個 ref 數(shù)據(jù)。

    • 第三部分就是返回響應(yīng)式數(shù)據(jù)對應(yīng)聲明的屬性對象

    這樣 toRef 就解決了響應(yīng)丟失的問題。

    2、toRefs 加入

    toRefs 就是將整個響應(yīng)式對象進(jìn)行解構(gòu)響應(yīng)化。實現(xiàn)代碼如下:

    function toRefs() {
        const ret = {}
        for (const key in obj) {
            ret[key] = toRef(obj,key)    
        }
        return ret
    }

    使用 for 循環(huán)逐一對屬性進(jìn)行轉(zhuǎn)換。這下我們再來看一下使用:

    const obj = reactive({foo:1,bar:2})
    const {foo,bar} = toRefs(obj)
    obj.foo++    // foo.value變?yōu)?了
    3、詭異的其它響應(yīng)式丟失情況

    當(dāng)屬性為非原始值的時候,解構(gòu)之后還是依然能響應(yīng)

    const obj = reactive({foo:{age:18},bar:2})
    const {foo,bar} = obj
    obj.foo.age++    // foo.age變?yōu)?了
    obj.bar++       // bar沒有改變,還是1

    這是為什么?原因其實很簡單,因為非原始值賦值的是引用地址,也就是說解構(gòu)后的變量其實還是指向原響應(yīng)式數(shù)據(jù)的屬性。而原始值就是單純的賦值,就不會進(jìn)行響應(yīng)。

    reactive 響應(yīng)數(shù)據(jù)重新賦值后不再響應(yīng),ref 響應(yīng)數(shù)據(jù)賦值后依然響應(yīng)
    let obj1 = reactive({foo:1,bar:2})
    let obj2 = ref({foo:1,bar:2})
     
    // 假如 obj1 與 obj2 直接展示在頁面上
    obj1 = {boo:2,far:3}    // 頁面 obj1 還是 {foo:1,bar:2}
    obj2.value = {boo:2,far:3}    // 頁面 obj2 變?yōu)?nbsp;{boo:2,far:3} 了

    這又是什么原因?reactive 重新賦值響應(yīng)丟失,就是重新賦值了新的對象,自然就成為普通數(shù)據(jù)了,不再響應(yīng)。而 ref 還是能響應(yīng),是因為 ref 在內(nèi)部進(jìn)行 set 處理。代碼如下:

    function ref(val){
        const wrapper = {
            value:val
            set value(val) {    // isObject 這里代表判斷是否是非原始值的一個方法
                 value = isObject(val) === 'Object' ? reactive(val) : val       
            }
        }
        Object.defineProperty(warpper,'__v_isRef',{
            value:true    
        })
        return reactive(wrapper)
    }

    我們明白了,其實 ref 在 set 中判斷了設(shè)置的新值是否是非原始值,如果是就調(diào)用 reactive 將其變?yōu)轫憫?yīng)式數(shù)據(jù)。

    四、unref 自動脫 ref

    我們在使用 ref 響應(yīng)式數(shù)據(jù)時,會覺得總是需要 .value 來獲取值,增加了用戶的心智負(fù)擔(dān)。

    那可不可以不通過 .value 訪問值,而時直接就能夠訪問值呢?

    這樣用于也不用關(guān)心某個數(shù)據(jù)到底是不是 ref 數(shù)據(jù),需不需要通過 value 屬性去獲取值。

    這就到了我們的 unref 出手了。unref 實現(xiàn)了自動脫 ref 能力,自動脫 ref 就是如果讀取的屬性是一個 ref,則直接將該 ref 對應(yīng)的 value 屬性值返回。

    我們來看一下 unref 的實現(xiàn):

    function unref(target) {
        return new Proxy(target,{
            get(target,key,receiver) {
                const value = Reflect.get(target,key,receiver)
                return value.__v_isRef ? value.value : value        
            },
            set(target,key,newValue,receiver) {
                const value = target[key]
                if (value.__v_isRef) {
                    value.value = newValue
                    return true            
                }        
                return Reflect.set(target,key,newValue,receiver)
            }
        })
    }

    我們發(fā)現(xiàn) unref 內(nèi)部使用 Proxy 代理了目標(biāo)對象,接收一個對象作為參數(shù),并返回該對象的代理對象。當(dāng)我們訪問 unref 的數(shù)據(jù)時,觸發(fā) get 捕獲器,然后再捕獲器內(nèi)部判斷了傳入對象是否是 ref 對象,如果是就直接返回 ref 的 .value 值。如果不是則直接返回代理對象。

    當(dāng)對 unref 返回的代理對象設(shè)置值時,觸發(fā) set 捕獲器,如果代理的對象時 ref ,就將需要設(shè)置的新值賦值給 .value,不是則直接進(jìn)行賦值處理。

    知識擴(kuò)展

    當(dāng)我們使用 ref 創(chuàng)建響應(yīng)式數(shù)據(jù)后,將其在模板中展示,為什么不用 .value 了。

    
    import { ref } from 'vue'
    const name = ref('小黑子')
    
     
    

    其實原因很簡單,在組件 setup 中聲明的 ref 響應(yīng)式數(shù)據(jù)會傳遞給 unref 函數(shù)進(jìn)行處理。所以在模板中訪問 ref 的值,無需通過 value 屬性來訪問。

    我們使用的 reactive 其實也是有 自動脫 ref 功能的,看一下下方例子:

    const count = ref(0)
    const obj = reactive({conut})
     
    obj.count    // 0

    我們可以看見 obj.count 是一個 ref 響應(yīng)式數(shù)據(jù)。在 count 外層包裹一層對象,再傳遞給 reactive 后,再訪問 obj.count 時就不需要再通過 value 屬性訪問值了。

    也正是因為 reactive 內(nèi)部也同樣實現(xiàn)了自動脫 ref 的能力。

    以上就是“vue3原始值響應(yīng)丟失如何解決”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家閱讀完這篇文章都有很大的收獲,小編每天都會為大家更新不同的知識,如果還想學(xué)習(xí)更多的知識,請關(guān)注創(chuàng)新互聯(lián)行業(yè)資訊頻道。


    網(wǎng)站題目:vue3原始值響應(yīng)丟失如何解決
    文章網(wǎng)址:http://weahome.cn/article/ghiccs.html

    其他資訊

    在線咨詢

    微信咨詢

    電話咨詢

    028-86922220(工作日)

    18980820575(7×24)

    提交需求

    返回頂部