Vue常見組件間通信方案及典型應(yīng)用場景詳解
什么是組件通信
所謂組件通信,就是組件之間的數(shù)據(jù)交互,也就是把一個組件A里面的數(shù)據(jù)傳遞到另一個組件B,并能夠讓組件B根據(jù)這個數(shù)據(jù)更新界面。
在 Vue中,可用的通信方案有很多,下面給大家描述幾個常用的組件通信方案及其典型的應(yīng)用場景。
1、父子組件通信場景
父子組件通信是最典型的組件通信場景,沒有之一,他們之間通信主要使用自定義屬性或者ref。
第一種是自定義屬性,使用較為廣泛,在子組件中使用 props 選項進行接收,子組件向父組件回傳數(shù)據(jù),使用自定義事件,在子組件中使用 $emit() 方法觸發(fā)執(zhí)行,并回傳數(shù)據(jù)給父組件。
// 定義組件h1-box Vue.component("h1-box", { // 通過props設(shè)置自定義屬性title接收組件外傳值 props: ["title"], // 通過emits設(shè)置自定義事件向外部發(fā)送事件 emits: ['give'], methods:{ clickHandler(){ // 通過$emit() 方法觸發(fā)執(zhí)行把當(dāng)前組件數(shù)據(jù)帶出去 this.$emit('give',{name:'lucy'}) } }, mounted() { // 通過this.title可以獲取外部傳入組件的值 console.log(this.title); }, });
第二種使用 ref 實現(xiàn)通信。我們知道,使用 ref 可以快速訪問一個組件實例對象及其內(nèi)部的數(shù)據(jù)和方法。這樣的話,在父組件中借助 refs可以訪問子級組件的數(shù)據(jù),還可以借助refs 可以訪問子級組件的數(shù)據(jù),還可以借助 refs可以訪問子級組件的數(shù)據(jù),還可以借助refs 調(diào)用子級組件中的方法,并傳遞事件參數(shù)給子級組件。當(dāng)然需要注意的是,在 Vue開發(fā)中,應(yīng)該盡量減少對 ref 的使用。
<div id="app"> <!-- 組件 --> <son ref="son"></son> </div> new Vue({ el: "#app", mounted() { // 獲取到組件實例,就可以任意操作組件實例 console.log(this.$refs.son); let vm2 = this.$refs.son; // 修改son組件的響應(yīng)式數(shù)據(jù)msg vm2.msg = "sz2111"; // 調(diào)用son組件的changeMsg方法 vm2.changeMsg("sz2116"); }, components: { // 定義組件son son: { data() { return { msg: "sz2114", }; }, methods: { changeMsg(val) { this.msg = val; }, }, template: `<h1>hello world</h1>`, }, }, });
2、兄弟組件通信場景
這種場景使用狀態(tài)提升,這個概念源自 React 中的狀態(tài)提升思想。
所謂狀態(tài)提升,就是當(dāng)兩個組件希望共享一個數(shù)據(jù)時,我們可以找到這兩個組件最近的父級組件,把這個要被共享的變量定義在最近的父組件中去,再通過 props 向下傳遞給子組件們。需要注意的是,狀態(tài)提升適合應(yīng)用在簡單的兄弟組件之間通信。當(dāng)遇到較為復(fù)雜的組件關(guān)系時,使用狀態(tài)提升就顯得麻煩了。
上圖圖中如果想點擊Font按鈕改變主題字體大小,也就是Title和Font兩個組件字體大小都改變。
如果Title和Font組件都維護自己組件的狀態(tài)數(shù)據(jù)就不太好傳遞數(shù)據(jù),就可以他他們的狀態(tài)存儲到共同的父組件狀態(tài)中,也就是狀態(tài)提升。
<!-- 父組件app --> <div id="app"> <!-- 子組件font --> <font :font="fontSize"></font> <!-- 子組件title --> <title :font="fontSize"></title> </div> // 父組件 new Vue({ el:'#app', data:{ // 把子組件共同需要的數(shù)據(jù)存儲在共同的父組件上 fontSize:12 } }) // 子組件 Vue.component('font',{ // 通過自定義屬性接受父組件傳入的值 props:['font'], template:`<div :style="{fontSize:font+'px'}">hello</div>` }) // 子組件 Vue.component('title',{ // 通過自定義屬性接受父組件傳入的值 props:['font'], template:`<div :style="{fontSize:font+'px'}">world</div>` })
3、根組件和后代組件通信場景
這種場景主要是provide/inject。使用 provide 選項,可以在任意組件中注入數(shù)據(jù);使用 inject 選項,可以在后代組件中接受父級組件注入的數(shù)據(jù)。需要注意的是,provide/inject 這種通信方案是沒有響應(yīng)式的,即父組件注入的數(shù)據(jù)發(fā)生變化時,后代組件不會自動更新
上圖在祖先組件通過provide傳入的數(shù)據(jù),在后代的所有組件中都可以通過inject獲取到。
下面是一個通過provide/inject傳值的例子
<div id="app"> <three></three> </div>
// 根組件 new Vue({ el: "#app", // provide用于在一個vue實例里面給后面的子孫實例傳遞數(shù)據(jù) provide: function () { return { a: 100, c: 300, d: this.obj, }; } }); // 定義組件one Vue.component("one", { // inject用于在子孫組件中接收祖先通過provide傳入的變量c inject: ["a"], template: ` <div> <h1>我是one組件---{{a}}</h1> </div> `, }); // 定義組件two,里面有子組件one Vue.component("two", { // inject用于在子孫組件中接收祖先通過provide傳入的變量c,并重名成myC inject: { myC: "c", }, template: ` <div> <h1>我是two組件---{{myC}}</h1> <one></one> </div> `, }); // 定義組件three,里面有子組件two Vue.component("three", { // inject用于在子孫組件中接收祖先通過provide傳入的變量b,并重名成bbb,同時設(shè)置默認值299999 inject: { bbb: { from: "b", default: 299999, }, }, template: ` <div> <h1>我是three組件---{{bbb}}</h1> <two></two> </div> `, });
4、插槽通信場景
在封裝組件時,可以為 組件添加自定義屬性。使用這個組件時,在父級組件中使用 #slotName='scope' 指令可以接收到子組件插槽傳遞過來的數(shù)據(jù)。ElementUI 中的 Table 表格、VantUI 中的 Tabbar 組件,都用到了插槽通信。
<div id="app"> <child> <template #abc> <h1>sz2114</h1> <h2>sz2115</h2> <h3>sz2116</h3> </template> <template #cindy> <h1>sz2014</h1> <h2>sz2015</h2> <h3>sz2016</h3> </template> </child> </div> // 全局組件 Vue.component("child", { template: ` <div> <slot name='cindy'></slot> <h1>hello world</h1> <slot name='abc'></slot> </div> `, }); let vm = new Vue({ el: "#app", });
5 無直接關(guān)系的組件通信場景
在沒有直接關(guān)聯(lián)的組件之間通信可以使用事件總線。事件總線是一種基于訂閱發(fā)布模式而設(shè)計的通信方案,在任意組件中訂閱指定“頻道”后,都能收到該“頻道”上的消息
。事件總線,它的強大之處在于:它是一種“一對多”的通信方案,還是一種“多頻道”的通信方案,非常強大。
<div id="app"> <input type="text" v-model="duanxin1" /> <button @click="clickHandler">給老師發(fā)消息</button> </div> // 中央事件總線: 類似一個事件對象 // 就像一個電信局 var bus = new Vue(); // 每個人就收短信都需要一個號碼 (事件名) // 想要接收短信要先去電信辦個卡 -- 注冊一個號碼 bus.$on("cyrevent", function (data) { console.log("短信內(nèi)容是"); console.log(data); }); new Vue({ el: "#app", data() { return { duanxin1: "", }; }, methods: { clickHandler() { // 別人想給我發(fā)消息 bus.$emit("cyrevent", this.duanxin1); }, }, });
6 大型項目中的復(fù)雜組件通信場景 - Vuex狀態(tài)管理
Vuex狀態(tài)管理是借助狀態(tài)管理工具,可以實現(xiàn)任意組件之間的數(shù)據(jù)通信。Vuex 提供了 state、mutations 等接口,可以方便地實現(xiàn)任意未知關(guān)系的組件之間的數(shù)據(jù)交互。因此,我們經(jīng)常稱 Vuex是 Vue開發(fā)中的終極通信方案。終極的意思,不是說它可以隨意地替代其它通信方案,而是說 Vuex很好用,能夠清晰地管理數(shù)據(jù)流。
上圖是vuex官方網(wǎng)站的工作流程圖,特別形象。
7 其他的一些組件通信方案
上面說到的是比較常用的組件通信方式,還有一些方式是上面通信方法不能使用的替代手段。
第一個是: $parent/$children
,借助這兩個 API,可以實現(xiàn)在組件樹之間任意穿梭。我們在當(dāng)前組件的作用域中,可以訪問到任意其它組件的內(nèi)部數(shù)據(jù),并調(diào)用它的方法。因此,這也是一種可用的通信方案。
第二個是:$attrs/$listeners
。使用 $attrs
可以訪問到父組件傳遞過來的自定義屬性(除 class 和 style 外),使用 $listeners
可以訪問并調(diào)用父組件傳遞過來的自定義事件,通過對自定義事件的調(diào)用還能向父組件回傳數(shù)據(jù)。這兩個內(nèi)置 API,在某種程度上可以看成是父子組件通信的替代方案。
總結(jié)
雖然 Vue中可用的通信方案很多,但要注意的是“別濫用”。在同一個項目中,選擇適合場景的通信方案很重要,不要使用過多的通信方式,這會導(dǎo)致代碼很難維護。一個數(shù)據(jù)流不清晰的 Web應(yīng)用,通常是很難得到持續(xù)發(fā)展的。
以上就是Vue常見組件間通信方案及典型應(yīng)用場景詳解的詳細內(nèi)容,更多關(guān)于Vue組件間通信應(yīng)用場景的資料請關(guān)注腳本之家其它相關(guān)文章!
相關(guān)文章
Vue是怎么渲染template內(nèi)的標(biāo)簽內(nèi)容的
這篇文章主要介紹了Vue是怎么渲染template內(nèi)的標(biāo)簽內(nèi)容的,文中通過示例代碼介紹的非常詳細,對大家的學(xué)習(xí)或者工作具有一定的參考學(xué)習(xí)價值,需要的朋友們下面隨著小編來一起學(xué)習(xí)學(xué)習(xí)吧2020-06-06詳解基于vue-router的動態(tài)權(quán)限控制實現(xiàn)方案
本篇文章主要介紹了詳解基于vue-router的動態(tài)權(quán)限實現(xiàn)方案,小編覺得挺不錯的,現(xiàn)在分享給大家,也給大家做個參考。一起跟隨小編過來看看吧2017-09-09vue element ui validate 主動觸發(fā)錯誤提示操作
這篇文章主要介紹了vue element ui validate 主動觸發(fā)錯誤提示操作,具有很好的參考價值,希望對大家有所幫助。一起跟隨小編過來看看吧2020-09-09vue + element動態(tài)多表頭與動態(tài)插槽
這篇文章主要介紹了vue + element動態(tài)多表頭與動態(tài)插槽,下面文章圍繞vue + element動態(tài)多表頭與動態(tài)插槽的相關(guān)資料展開文章的內(nèi)容,具有一定的參考價值,需要的小伙伴可以參考一下,希望對大家有所幫助2021-12-12