亚洲精品久久久中文字幕-亚洲精品久久片久久-亚洲精品久久青草-亚洲精品久久婷婷爱久久婷婷-亚洲精品久久午夜香蕉

您的位置:首頁(yè)技術(shù)文章
文章詳情頁(yè)

Vue 數(shù)據(jù)響應(yīng)式相關(guān)總結(jié)

瀏覽:3日期:2022-10-08 11:25:59

在說(shuō)數(shù)據(jù)響應(yīng)式之前,我們要解決一個(gè)很重要的問(wèn)題,那就是Vue到底對(duì)data做了什么?先從getter和setter說(shuō)起,我們用那個(gè)他們來(lái)對(duì)虛擬的屬性進(jìn)行讀寫(xiě)。

getter和setter

有如下代碼

let obj0 = { 姓: '高', 名: '圓圓', age: 18};// 需求一,得到姓名let obj1 = { 姓: '高', 名: '圓圓', 姓名() { return this.姓 + this.名; }, age: 18};console.log('需求一:' + obj1.姓名());//高圓圓

此時(shí)我們log出來(lái)的結(jié)果是高圓圓,這個(gè)大家都能看懂,但是姓名后面的括號(hào)能刪掉嗎?不能,因?yàn)樗呛瘮?shù),那么我們?cè)趺慈サ衾ㄌ?hào)呢?下面就有我們的需求二

// 需求二,姓名不要括號(hào)也能得出值let obj2 = { 姓: '高', 名: '圓圓', get 姓名() { return this.姓 + this.名; }, age: 18};console.log('需求二:' + obj2.姓名);//高圓圓

此時(shí)我們使用getter ,不加括號(hào)也能得出值。那么我們要怎么改變這個(gè)名字呢?

// 需求三:姓名可以被寫(xiě)let obj3 = { 姓: '高', 名: '圓圓', get 姓名() { return this.姓 + this.名; }, set 姓名(xxx){ this.姓 = xxx[0] this.名 = xxx.slice(1) }, age: 18};obj3.姓名 = ’高媛媛’console.log(`需求三:姓 ${obj3.姓},名 ${obj3.名}`)//高媛媛

有g(shù)et就有set,setter就是這樣用的。我們用 屬性值 = xxx 觸發(fā) set 函數(shù),姓名就可以被寫(xiě)啦。但是我們?cè)谛枨笕写虺?console.log(obj3) 會(huì)得到如下圖所示:

Vue 數(shù)據(jù)響應(yīng)式相關(guān)總結(jié)

如圖為什么會(huì)顯示 姓名:(...) 呢? 這其實(shí)是一個(gè)get set,瀏覽器在顯示這個(gè)姓名的時(shí)候就打印出 姓名:(...) ,這說(shuō)明我們可以在需求三中對(duì)姓名進(jìn)行讀和寫(xiě),但是并不存在一個(gè)叫做姓名的屬性,而是有g(shù)et和set來(lái)模擬對(duì)姓名進(jìn)行的操作。

Object.defineProperty

在如上例子中,我們?cè)诙x對(duì)象的時(shí)候就直接使用get和set,但是如果對(duì)象已經(jīng)被聲明完了,那我們?cè)趺蠢^續(xù)加上get呢?我們就要用到Object.defineProperty,還是需求三,我們加入如下代碼就可以在定義完之后再加get和set了:

var _xxx = 0Object.defineProperty(obj3,’xxx’,{ get(){ return _xxx }, set(value){ _xxx= value }})

接下來(lái)我們就可以解決一開(kāi)始的問(wèn)題了:Vue到底對(duì)data做了什么?我們舉幾個(gè)例子看看:

let data0 = { n: 0}

先聲明一個(gè)data0,需求一:用 Object.defineProperty 定義 n:

let data1 = {}Object.defineProperty(data1, ’n’, { value: 0})console.log(`需求一:${data1.n}`)//需求一:0

需求二:n 不能小于 0:

let data2 = {}data2._n = 0 // _n 用來(lái)偷偷存儲(chǔ) n 的值,默認(rèn)為0Object.defineProperty(data2, ’n’, { get(){ return this._n }, set(value){ if(value < 0) return this._n = value }})console.log(`需求二:${data2.n}`)//0data2.n = -1console.log(`需求二:${data2.n} 設(shè)置為 -1 失敗`)//0設(shè)置為 -1 失敗data2.n = 1console.log(`需求二:${data2.n} 設(shè)置為 1 成功`)//0設(shè)置為 1 成功

可是如果對(duì)方直接使用data2._n呢?我們能不能做到不在對(duì)象上暴露任何能夠被訪問(wèn)的東西呢?這時(shí)候我們就要使用代理:

let data3 = proxy({ data:{n:0} }) // 括號(hào)里是匿名對(duì)象,無(wú)法訪問(wèn)function proxy({data}){ const obj = {} // 這里的 ’n’ 寫(xiě)死了,理論上應(yīng)該遍歷 data 的所有 key,這里做了簡(jiǎn)化 // 因?yàn)槲遗履銈兛床欢?Object.defineProperty(obj, ’n’, { get(){ return data.n }, set(value){ if(value<0)return data.n = value } }) return obj // obj 就是代理}// data3 就是 objconsole.log(`需求三:${data3.n}`)data3.n = -1console.log(`需求三:${data3.n},設(shè)置為 -1 失敗`)data3.n = 1console.log(`需求三:${data3.n},設(shè)置為 1 成功`)

可是如果不想用代理,要怎么做呢?

let myData = {n:0}let data4 = proxy({ data:myData }) // 括號(hào)里是匿名對(duì)象,無(wú)法訪問(wèn)// data3 就是 objconsole.log(`杠精:${data4.n}`)//0myData.n = -1console.log(`杠精:${data4.n},設(shè)置為 -1 失敗了嗎!?`)

現(xiàn)在這樣還是能更改myData,所以我們又有一個(gè)需求:就算是用戶擅自修改myData,也要攔截:

let myData5 = {n:0}let data5 = proxy2({ data:myData5 }) // 括號(hào)里是匿名對(duì)象,無(wú)法訪問(wèn)function proxy2({data}){ // 這里的 ’n’ 寫(xiě)死了,理論上應(yīng)該遍歷 data 的所有 key,這里做了簡(jiǎn)化 let value = data.n//保存開(kāi)始的n Object.defineProperty(data, ’n’, {//聲明一個(gè)新的n get(){ return value }, set(newValue){ if(newValue<0)return value = newValue } })

就加了上面幾句,這幾句話會(huì)監(jiān)聽(tīng) data

const obj = {} Object.defineProperty(obj, ’n’, { get(){ return data.n }, set(value){ if(value<0)return//這句話多余了 data.n = value } }) return obj // obj 就是代理}// data3 就是 objconsole.log(`需求五:${data5.n}`)//0myData5.n = -1console.log(`需求五:${data5.n},設(shè)置為 -1 失敗了`)//0myData5.n = 1console.log(`需求五:${data5.n},設(shè)置為 1 成功了`)//1

當(dāng)我們寫(xiě)vm = new Vue({data:myData})時(shí),Vue做了兩件事情:

讓vm成為myData的代理(proxy),可以通過(guò)this訪問(wèn)vm 會(huì)對(duì)myData所有的屬性進(jìn)行監(jiān)控,為了防止myData的屬性變了,vm卻不知道,知道了屬性變化之后就可以調(diào)用render(data),UI就可以自動(dòng)刷新

那么我們就可以回到標(biāo)題了,什么是數(shù)據(jù)響應(yīng)式呢?如果一個(gè)物體能夠?qū)ν饨绲拇碳ぷ龀龇磻?yīng),那么它就是響應(yīng)式的。Vue的data是響應(yīng)式的,const vm = new Vue({data:{n:0}})在這個(gè)代碼中如果修改vm.n那么UI中的n就會(huì)做出相應(yīng)的更新,Vue通過(guò)Object.defineProperty來(lái)實(shí)現(xiàn)數(shù)據(jù)響應(yīng)式。響應(yīng)式網(wǎng)頁(yè)又是什么呢?即如果改變窗口的大小,網(wǎng)頁(yè)內(nèi)容會(huì)做出相應(yīng)的改變,那么這個(gè)網(wǎng)頁(yè)就叫響應(yīng)式網(wǎng)頁(yè)。

以上就是Vue 數(shù)據(jù)響應(yīng)式相關(guān)總結(jié)的詳細(xì)內(nèi)容,更多關(guān)于Vue 數(shù)據(jù)響應(yīng)式的資料請(qǐng)關(guān)注好吧啦網(wǎng)其它相關(guān)文章!

標(biāo)簽: Vue
相關(guān)文章:
主站蜘蛛池模板: 国产精品一区二区久久 | 国产精品久久久久久一级毛片 | 欧洲免费无线码二区5 | 香蕉久久久久 | 日韩欧美国产中文 | 欧美嗯啊 | 亚洲国产剧情 | 成人国内精品久久久久影 | 午夜影院在线观看视频 | 日本黄色一级片视频 | 在线一区免费播放 | 亚洲一区二区欧美日韩 | 日韩欧美a级高清毛片 | 91老女人| 亚洲欧美日韩中文字幕久久 | 九九爱这里只有精品 | 婷婷亚洲天堂 | 可以直接看的av网址站 | 三级福利片| 欧美视频三区 | 欧美一级毛片欧美大尺度一级毛片 | 麻豆精品视频网站在线观看 | 欧美在线视 | 看黄网址在线观看 | 久久精品久久精品久久 | 国产在线不卡午夜精品2021 | 亚洲人成网站在线在线 | 国产99网站 | 野战好大好紧好爽视频 | 日本69色视频在线观看 | 99re这里有免费视频精品 | 亚洲一区二区色 | 午夜网站在线播放 | 免费中日高清无专码有限公司 | 高清国产天干天干天干不卡顿 | 草草网 | 国产又黄又爽又色视频影视网免费 | 亚欧在线视频 | 日本免费人成黄页在线观看视频 | 成人影片在线播放 | 国产在线观看高清精品 |