這篇文章給大家介紹如何在Vue中使用vm.$attrs,內(nèi)容非常詳細(xì),感興趣的小伙伴們可以參考借鑒,希望對大家能有所幫助。
成都創(chuàng)新互聯(lián)主營靖西網(wǎng)站建設(shè)的網(wǎng)絡(luò)公司,主營網(wǎng)站建設(shè)方案,app軟件開發(fā),靖西h5成都小程序開發(fā)搭建,靖西網(wǎng)站營銷推廣歡迎靖西等地區(qū)企業(yè)咨詢
1、vm.$attrs簡介
包含了父作用域中不作為 prop 被識別 (且獲取) 的特性綁定 (class 和 style 除外)。當(dāng)一個組件沒有聲明任何 prop 時,這里會包含所有父作用域的綁定 (class 和 style 除外),并且可以通過 v-bind="$attrs" 傳入內(nèi)部組件——在創(chuàng)建更高層次的組件時非常有用。
猛一看有點看不明白....
2、場景介紹
vue中一個比較令人煩惱的事情是屬性只能從父組件傳遞給子組件。這也就意味著當(dāng)你想向嵌套層級比較深組件數(shù)據(jù)傳遞,只能由父組件傳遞給子組件,子組件再傳遞給孫子組件...像下面這樣:
....
就這樣一層一層的往下傳遞passdown這個變量,最后才能用{{passdown}}。
假如我們需要傳遞的屬性只有1,2個還行,但是如果我們要傳遞的有幾個或者10來個的情況,這會是什么樣的場景,我們會在每個組件不停的props,每個必須寫很多遍。有沒有其它方便的寫法?有,通過vuex的父子組件通信,的確這個是一個方法,但是還有其它的方法,這個就是我們要說的。通過inheritAttrs選項,以及實例屬性$attrs
3、實例:
上邊的代碼,我們在組件里只是用了title這個屬性,massgae屬性我么是沒有用的,那么下瀏覽器渲染出來是什么樣呢?如下圖:
我們看到:組件內(nèi)未被注冊的屬性將作為普通html元素屬性被渲染,如果想讓屬性能夠向下傳遞,即使prop組件沒有被使用,你也需要在組件上注冊。這樣做會使組件預(yù)期功能變得模糊不清,同時也難以維護組件的DRY。在Vue2.4.0,可以在組件定義中添加inheritAttrs:false,組件將不會把未被注冊的props呈現(xiàn)為普通的HTML屬性。但是在組件里我們可以通過其$attrs可以獲取到?jīng)]有使用的注冊屬性,如果需要,我們在這也可以往下繼續(xù)傳遞。
如果我們在子組件里設(shè)置 inheritAttrs: false:
components:{ 'mytest':{ template:`這是個h2標(biāo)題{{title}}`, props:['title'], inheritAttrs: false, data(){ return{ mag:'111' } }, created:function(){ console.log(this.$attrs)//注意這里 } }
渲染效果如下:
不繼承的情況.png
補充:說一下$attrs的使用
有一個頁面由父組件,子組件,孫子組件構(gòu)成,如下:
上面的代碼在頁面的效果是如下圖
如果attrs被綁定在子組件childcom上后,我們就可以在孫子組件grandcom里獲取到this.$attrs的值。這個{{$attrs}}的值是父組件中傳遞下來的props(除了子組件childcom組件中props聲明的)。
記住孫子組件grandcom里獲取到this.$attrs的值是除了子組件childcom聲明的元素!記住是除了子組件childcom聲明的元素!例如上面的代碼我在子組件childcom組件的props里聲明了name,那么我在孫子組件grandcom里獲取到的$attrs就不包含name屬性,那么this.$attrs = { 'age':'30', 'sex':'男'}。
說一下$attrs的優(yōu)勢到底在哪
假如我們要做一個頁面,有父組件,子組件,孫子組件,如下:
如上代碼,假如我想在子組件想獲取到父組件的name屬性值,在孫子組件獲取父組件的age屬性值,用props的話就必須在父組件把name和age的值通過props傳遞到子組件,子組件在通過props把age的值傳遞到孫子組件,到這里看明白了吧,孫子組件需要的age在子組件里沒有用到,但是為了能讓孫子組件獲取到,你必須從父組件 傳到子組件,在在子組件傳遞到孫子組件。
但是用$attrs就不用那么麻煩,如下:
子組件綁定了"$attrs",孫子組件就能獲取到除了name屬性外所有由父組件傳遞下來的屬性。如果孫子組件也想獲取到name屬性那么,在綁定個name如下,
細(xì)細(xì)體會下是不是這個道理。實在不行的話敲一敲代碼自己試驗下,你就會豁然開朗。
補充一下:inheritAttrs屬性
關(guān)于inheritAttrs這個屬性跟獲取到$attrs的值沒有關(guān)系,inheritAttrs通常在編寫基礎(chǔ)組件時候會用到。官網(wǎng)原話:默認(rèn)情況下父作用域的不被認(rèn)作 props 的特性綁定 (attribute bindings) 將會“回退”且作為普通的 HTML 特性應(yīng)用在子組件的根元素上。當(dāng)撰寫包裹一個目標(biāo)元素或另一個組件的組件時,這可能不會總是符合預(yù)期行為。通過設(shè)置 inheritAttrs 到 false,這些默認(rèn)行為將會被去掉。而通過 (同樣是 2.4 新增的) 實例屬性 $attrs 可以讓這些特性生效,且可以通過 v-bind 顯性的綁定到非根元素上。
注意:這個選項不影響 class 和 style 綁定。
在Vue2.4.0之前版本,組件內(nèi)未被注冊的屬性將作為普通html元素屬性被渲染。
inheritAttrs到底有啥用?到底用在哪里?看下邊代碼,
上面代碼你覺得input上會怎么顯示? 父組件傳遞了type="text",子組件里input 上type="number",那渲染到頁面會是什么樣?渲染圖如下:
默認(rèn)情況.png
看到?jīng)],父組件傳遞的type="text"覆蓋了input 上type="number",這豈不是把我的input數(shù)據(jù)類型都給改變了,這豈不是有問題,這不是我想要的!?。?!看到這里明白了嗎?回頭去體會下上面官網(wǎng)的原話?。。?/p>
需求:我需要input 上type="number"類型不變,但是我還是要取到父組件的type="text"的值,那么代碼如下:
頁面渲染圖如下:
需求.png
到這,我想大家都明白了inheritAttrs的作用了吧。默認(rèn)情況下vue會把父作用域的不被認(rèn)作 props 的特性綁定 且作為普通的 HTML 特性應(yīng)用在子組件的根元素上。綁定就綁定,顯示就顯示,沒啥大不了的,但是怕就怕遇到一些特殊的,就比如上面的input的情況,這個時候inheritAttrs:false的作用就出來啦。
順道補充一下:$listeners
父組件-子組件-孫子組件,,,,現(xiàn)在我要你在孫子組件里改變父組件的值,怎么改?有很多方法啦,但是$listeners給我們提供了一個新的思路。話不多說,直接上代碼
頁面渲染如下:
關(guān)于如何在Vue中使用vm.$attrs就分享到這里了,希望以上內(nèi)容可以對大家有一定的幫助,可以學(xué)到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。