本篇內(nèi)容介紹了“vue組件間如何進行通信”的有關知識,在實際案例的操作過程中,不少人都會遇到這樣的困境,接下來就讓小編帶領大家學習一下如何處理這些情況吧!希望大家仔細閱讀,能夠?qū)W有所成!
成都創(chuàng)新互聯(lián)公司是一家專業(yè)提供進賢企業(yè)網(wǎng)站建設,專注與網(wǎng)站制作、成都網(wǎng)站設計、HTML5、小程序制作等業(yè)務。10年已為進賢眾多企業(yè)、政府機構(gòu)等服務。創(chuàng)新互聯(lián)專業(yè)網(wǎng)站設計公司優(yōu)惠進行中。
vue是數(shù)據(jù)驅(qū)動視圖更新的框架, 我們平時開發(fā),都會把頁面不同模塊拆分成一個一個vue組件, 所以對于vue來說組件間的數(shù)據(jù)通信非常重要,那么組件之間如何進行數(shù)據(jù)通信的呢?
首先我們需要知道在vue中組件之間存在什么樣的關系, 才更容易理解他們的通信方式。
一般我們分為如下關系:
父子組件之間通信
非父子組件之間通信(兄弟組件、隔代關系組件等)
父組件通過props
的方式向子組件傳遞數(shù)據(jù),而通過$emit
子組件可以向父組件通信。
父組件向子組件傳值
{{ msg }}
注意:
prop 只可以從上一級組件傳遞到下一級組件(父子組件),即所謂的單向數(shù)據(jù)流。而且 prop 只讀,不可被修改,所有修改都會失效并警告。
第一,不應該在一個子組件內(nèi)部改變 prop,這樣會破壞單向的數(shù)據(jù)綁定,導致數(shù)據(jù)流難以理解。如果有這樣的需要,可以通過 data 屬性接收或使用 computed
屬性進行轉(zhuǎn)換。
第二,如果 props
傳遞的是引用類型(對象或者數(shù)組)
,在子組件中改變這個對象或數(shù)組,父組件的狀態(tài)會也會做相應的更新,利用這一點就能夠?qū)崿F(xiàn)父子組件數(shù)據(jù)的“雙向綁定”
,雖然這樣實現(xiàn)能夠節(jié)省代碼,但會犧牲數(shù)據(jù)流向的簡潔性
,令人難以理解,最好不要這樣去做。
想要實現(xiàn)父子組件的數(shù)據(jù)“雙向綁定”,可以使用 v-model
或 .sync
。
子組件向父組件傳值
使用 $emit
向父組件傳數(shù)據(jù),原理這樣的: 父組件在子組件通過v-on
監(jiān)聽函數(shù)并接收參數(shù),vue框架就在子組件監(jiān)聽了你v-on="fn"
的fn事件函數(shù),在子組件使用$emit
就能觸發(fā)了,下面寫個例子。
{{ msg }}
v-model
是用來在表單控件
或者組件
上創(chuàng)建雙向綁定
的,他的本質(zhì)是 v-bind
和 v-on
的語法糖
,在一個組件上使用 v-model
,默認會為組件綁定名為 value 的 prop
和名為 input
的事件。
當我們組件中的某一個 prop
需要實現(xiàn)上面所說的”雙向綁定“時,v-model
就能大顯身手了。有了它,就不需要自己手動在組件上綁定監(jiān)聽當前實例上的自定義事件,會使代碼更簡潔
。
下面以一個 input 組件實現(xiàn)的核心代碼,介紹下 v-model
的應用。
上面例子看到,v-model="inputValue"
他的本質(zhì)就是 v-bind 和 v-on 的語法糖
,默認為父組件綁定名為 :value="inputValue"
的屬性,和@input="(v) => { this.inputValue = v }"
事件,子組件通過 this.$emit('input', value)
通知父組件
所以他原理也是利用了我們上面講的父子組件傳參 props / $emit
方式來實現(xiàn)雙向綁定
有時,在某些特定的控件中名為 value 的屬性會有特殊的含義,這時可以通過 v-model
選項來回避這種沖突。
.sync
修飾符在 vue 1.x 的版本中就已經(jīng)提供,1.x 版本中,當子組件改變了一個帶有 .sync
的 prop
的值時,會將這個值同步到父組件中的值。這樣使用起來十分方便,但問題也十分明顯,這樣破壞了單向數(shù)據(jù)流,當應用復雜時,debug 的成本會非常高。
于是乎,在vue 2.0中移除了 .sync
。但是在實際的應用中,.sync
是有它的應用場景的,所以在 vue 2.3
版本中,又迎來了全新的 .sync
。
新的 .sync
修飾符所實現(xiàn)的已經(jīng)不再是真正的雙向綁定,它的本質(zhì)和 v-model
類似,只是一種縮寫。
正常封裝組件例子:
上面的代碼,使用 .sync
就可以寫成
這樣,在子組件中,就可以通過下面代碼來實現(xiàn)對這個 prop 重新賦值了。
this.$emit('update:title', newTitle)
看到這里,是不是發(fā)現(xiàn) .sync
修飾符 和 v-model
很相似,也是語法糖, v-bind:title.sync
也就是 等效于 v-bind:title="doc.title" v-on:update:title="doc.title = $event"
.sync
從功能上看和 v-model
十分相似,都是為了實現(xiàn)數(shù)據(jù)的“雙向綁定”
,本質(zhì)上,也都不是真正的雙向綁定,而是語法糖
。
相比較之下,.sync
更加靈活,它可以給多個 prop
使用,而 v-model
在一個組件中只能有一個。
從語義上來看,v-model
綁定的值是指這個組件的綁定值,比如 input 組件
,select 組件
,日期時間選擇組件
,顏色選擇器組件
,這些組件所綁定的值使用 v-model
比較合適。其他情況,沒有這種語義,個人認為使用 .sync
更好。
通過$parent
和$children
就可以訪問組件的實例,拿到實例代表什么?代表可以訪問此組件的所有方法
和data
。列子如下:
{{msg}}
{{messageA}}獲取父組件的值為: {{parentVal}}
要注意邊界情況,如在#app
上拿$parent
得到的是new Vue()
的實例,在這實例上再拿$parent
得到的是undefined
,而在最底層的子組件拿$children
是個空數(shù)組
。也要注意得到parent和children的值不一樣,$children 的值是數(shù)組
,而$parent是個對象
props $emit
、 $parent $children
兩種方式用于父子組件之間的通信, 而使用props進行父子組件通信更加普遍,二者皆不能用于非父子組件之間的通信。
provide / inject
是vue2.2.0新增的api, 簡單來說就是父組件中通過provide來提供變量
, 然后再子組件中通過inject來注入變量
。
官方描述: 這對選項需要一起使用,以允許一個祖先組件向其所有子孫后代注入一個依賴,不論組件層次有多深,并在其上下游關系成立的時間里始終生效
provide
選項應該是
一個對象或返回一個對象的函數(shù)。該對象包含可注入其子孫的屬性。在該對象中你可以使用 ES2015 Symbols 作為 key,但是只在原生支持 Symbol 和 Reflect.ownKeys 的環(huán)境下可工作。
inject
選項應該是:
一個字符串數(shù)組
一個對象(詳情點擊這里)
// 祖先組件 提供foo //第一種 export default { name: "father", provide() { return { foo: 'hello' } }, } //第二種 export default { name: "father", provide: { foo:'hello~~~~' }, }
//后代組件 注入foo, 直接當做this.foo來用 export default { inject:['foo'], }
上面的兩種用法有什么區(qū)別嗎?
如果你只是傳一個字符串,像上面的hello
,那么是沒有區(qū)別的,后代都能讀到。
如果你需要this
對象屬性的值(如下所示代碼),那么第二種是傳不了的,后代組件拿不到數(shù)據(jù)。所以建議只寫第一種
//當你傳遞對象給后代時 provide() { return { test: this.msg } },
注意:一旦注入了某個數(shù)據(jù),比如上面示例中的 foo,那這個組件中就不能再聲明 foo 這個數(shù)據(jù)了,因為它已經(jīng)被父級占有。
這是刻意為之的。然而,如果你傳入了一個可監(jiān)聽的對象
,那么其對象的屬性還是可響應的。因為對象是引用類型。
先來個值類型的數(shù)據(jù)(也就是字符串)例子,不會響應
provide(){ return{ test:this.msg } }, data() { return { msg: "Welcome to Your Vue.js App", } } mounted(){ setTimeout(()=>{ this.msg = "halo world"; console.log(this._provided.msg) //log:Welcome to Your Vue.js App },3000) },
如上所示,這樣做是不行的,打印出來的 _provided
中的數(shù)據(jù)并沒有改,子組件取得值也沒變。
你甚至可以直接給 this._provided.msg
賦值,但是即使是_provided.msg 里面的值改變了,子組件的取值,依然沒有變。
當你的參數(shù)是對象的時候,就可以響應了,如下:
provide(){ return{ test:this.activeData } }, data() { return { activeData:{name:'halo'}, } } mounted(){ setTimeout(()=>{ this.activeData.name = 'world'; },3000) }
這就是vue官方中寫道的對象的屬性是可以響應的
provide/inject不是只能從祖先傳遞給后代嗎?是的,但是,如果我們綁定到最頂層的組件app.vue,是不是所有后代都接收到了,就是當做全局變量來用了。
//app.vue export default { name: 'App', provide(){ return{ app:this } }, data(){ return{ text:"it's hard to tell the night time from the day" } }, methods:{ say(){ console.log("Desperado, why don't you come to your senses?") } } }
//其他所有子組件,需要全局變量的,只需要按需注入app即可 export default { inject:['foo','app'], mounted(){ console.log(this.app.text); // 獲取app中的變量 this.app.say(); // 可以執(zhí)行app中的方法,變身為全局方法! } }
用vue-router
重新路由到當前頁面,頁面是不進行刷新的
采用window.reload()
,或者router.go(0)
刷新時,整個瀏覽器進行了重新加載,閃爍,體驗不好
那我們怎么做呢?
跟上面的原理差不多,我們只在控制路由的組件中寫一個函數(shù)(使用v-if控制router-view的顯示隱藏,這里的原理不作贅述),然后把這個函數(shù)傳遞給后代,然后在后代組件中調(diào)用這個方法即可刷新路由啦。
//app.vueexport default { name: 'App', provide() { return { reload: this.reload } }, data() { return { isShowRouter: true, } }, methods:{ reload() { this.isShowRouter = false; this.$nextTick(() => { this.isShowRouter = true; }) } } }
//后代組件 export default { inject: ['reload'], }
這里 provide 使用了函數(shù)傳遞給后代,然后后代調(diào)用這個函數(shù),這種思路,也是可以做子后代向父組件傳參通訊的思路了。這里的原理,和 event 事件訂閱發(fā)布就很像了
ref
:如果在普通的 DOM 元素上使用,引用指向的就是 DOM 元素
;如果用在子組件
上,引用就指向組件實例
,可以通過實例直接調(diào)用組件的方法或訪問數(shù)據(jù), 我們看一個ref 來訪問組件的例子:
// 子組件 A.vue export default { data () { return { name: 'Vue.js' } }, methods: { sayHello () { console.log('hello') } } }
// 父組件 app.vue
ref 這種方式,就是獲取子組件的實例,然后可以直接子組件的方法和訪問操作data的數(shù)據(jù),就是父組件控制子組件的一種方式,子組件想向父組件傳參或操作,只能通過其他的方式了
eventBus
呢,其實原理就是 事件訂閱發(fā)布,eventBus
又稱為事件總線,在vue中可以使用它來作為溝通橋梁的概念, 就像是所有組件共用相同的事件中心,可以向該中心注冊發(fā)送事件或接收事件, 所以組件都可以通知其他組件。
這里我們可以直接使用 vue 自帶的事件監(jiān)聽,也就是 emiton,我們來簡單封裝下:
首先需要創(chuàng)建一個事件總線并將其導出, 以便其他模塊可以使用或者監(jiān)聽它.
新建一個 event-bus.js
文件
// event-bus.js import Vue from 'vue' export const EventBus = new Vue()
發(fā)生事件
假設你有兩個組件: additionNum 和 showNum, 這兩個組件可以是兄弟組件也可以是父子組件;這里我們以兄弟組件為例:
// addtionNum.vue 中發(fā)送事件
接收事件
// showNum.vue 中接收事件計算和: {{count}}
這樣就實現(xiàn)了在組件addtionNum.vue
中點擊相加按鈕, 在showNum.vue中利用傳遞來的 num 展示求和的結(jié)果.
移除事件監(jiān)聽者
如果想移除事件的監(jiān)聽, 可以像下面這樣操作:
import { eventBus } from 'event-bus.js' EventBus.$off('addition')
這里使用自己封裝一套eventBus也行,方便自己想干啥就干啥, 下面貼封裝好的一套給大家
/* eslint-disable no-console */ // 事件映射表 let eventMap = {} /** * 監(jiān)聽事件 * @param {string} eventName 事件名 * @param {function} listener 回調(diào)函數(shù) * @param {object} instance 注冊事件的實例 */ function on(eventName, listener, instance) { eventMap[eventName] = eventMap[eventName] || [] eventMap[eventName].push({ listener, instance, }) } // 監(jiān)聽事件,只執(zhí)行一次 function once(eventName, listener, instance) { eventMap[eventName] = eventMap[eventName] || [] eventMap[eventName].push({ listener, instance, once: true, }) } // 解除事件監(jiān)聽 function off(eventName, listener) { // 解除所有事件監(jiān)聽 if (!eventName) { eventMap = {} return } // 沒有對應事件 if (!eventMap[eventName]) { return } // 解除某事件監(jiān)聽 eventMap[eventName].forEach((currentEvent, index) => { if (currentEvent.listener === listener) { eventMap[eventName].splice(index, 1) } }) } // 發(fā)送事件,執(zhí)行對應響應函數(shù) function emit(eventName, ...args) { if (!eventMap[eventName]) { return } eventMap[eventName].forEach((currentEvent, index) => { currentEvent.listener(...args) if (currentEvent.once) { eventMap[eventName].splice(index, 1) } }) } // 顯示當前注冊的事件,代碼優(yōu)化時使用 function showEventMap(targetEventName) { if (targetEventName) { // 查看具體某個事件的監(jiān)聽情況 eventMap[targetEventName].forEach(eventItem => { console.log(targetEventName, eventItem.instance, eventItem.listener) }) } else { // 查看所以事件的監(jiān)聽情況 Object.keys(eventMap).forEach(eventName => { eventMap[eventName].forEach(eventItem => { console.log(eventName, eventItem.instance, eventItem.listener) }) }) } } // 提供 vue mixin 方法,在 beforeDestroy 自動注銷事件監(jiān)聽 export const mixin = { created() { // 重載 on 函數(shù),收集本組件監(jiān)聽的事件,待消除時,銷毀事件監(jiān)聽 this.$eventListenerList = [] this.$event = { off, once, emit, showEventMap } this.$event.on = (eventName, listener) => { this.$eventListenerList.push({ eventName, listener }) on(eventName, listener) } }, // 消除組件時,自動銷毀事件監(jiān)聽 beforeDestroy() { this.$eventListenerList.forEach(currentEvent => { off(currentEvent.eventName, currentEvent.listener) }) }, } export default { on, off, once, emit, showEventMap }
如何使用呢,只需在 項目的 main.js
, 引入 ,然后 Vue.mixin 即可,如下:
// main.js import Vue from 'vue' import { mixin as eventMixin } from '@/event/index' Vue.mixin(eventMixin)
在vue項目其他文件,就可以直接 this.$event.on
this.$event.$emit
如下:
this.$event.on('test', (v) => { console.log(v) }) this.$event.$emit('test', 1)
還順便封裝了個mixin, 好處呢,就是在vue頁面監(jiān)聽事件后,頁面銷毀后,也自動銷毀了事件監(jiān)聽
Vuex
是一個專為 Vue.js 應用程序開發(fā)的狀態(tài)管理模式
。它采用集中式存儲管理應用的所有組件的狀態(tài),并以相應的規(guī)則保證狀態(tài)以一種可預測的方式發(fā)生變化.
Vuex
解決了多個視圖依賴于同一狀態(tài)和來自不同視圖的行為需要變更同一狀態(tài)的問題,將開發(fā)者的精力聚焦于數(shù)據(jù)的更新而不是數(shù)據(jù)在組件之間的傳遞上
state
:用于數(shù)據(jù)的存儲,是store中的唯一數(shù)據(jù)源
getters
:如vue中的計算屬性一樣,基于state數(shù)據(jù)的二次包裝,常用于數(shù)據(jù)的篩選和多個數(shù)據(jù)的相關性計算
mutations
:類似函數(shù),改變state數(shù)據(jù)的唯一途徑,且不能用于處理異步事件
actions
:類似于mutation,用于提交mutation來改變狀態(tài),而不直接變更狀態(tài),可以包含任意異步操作
modules
:類似于命名空間,用于項目中將各個模塊的狀態(tài)分開定義和操作,便于維護
這里我們先新建 store文件夾
, 對Vuex進行一些封裝處理
在 store 文件夾下添加 index.js
文件
// index.js // 自動掛載指定目錄下的store import Vue from 'vue' import Vuex from 'vuex' Vue.use(Vuex) let modules = {} // @/store/module 目錄下的文件自動掛載為 store 模塊 const subModuleList = require.context('@/store/modules', false, /.js$/) subModuleList.keys().forEach(subRouter => { const moduleName = subRouter.substring(2, subRouter.length - 3) modules[moduleName] = subModuleList(subRouter).default }) export default new Vuex.Store({ state: {}, mutations: {}, actions: {}, modules })
在 store 文件夾下添加 module
文件夾,在module文件夾再新建 user.js
文件
// user.js import user from '@/utils/user.js' import userApi from '@/apis/user' import { OPEN_ACCOUNT_STAGE, STAGE_STATUS } from '@/constant' let getUserPromise = null export default { namespaced: true, state() { return { userInfo: null, // 用戶信息 isLogined: !!user.getToken(), // 是否已經(jīng)登錄 } }, mutations: { // 更新用戶信息 updateUser(state, payload) { state.isLogined = !!payload state.userInfo = payload }, }, actions: { // 獲取當前用戶信息 async getUserInfo(context, payload) { // forceUpdate 表示是否強制更新 if (context.state.userInfo && !payload?.forceUpdate) { return context.state.userInfo } if (!getUserPromise || payload?.forceUpdate) { getUserPromise = userApi.getUserInfo() } // 獲取用戶信息 try { const userInfo = await getUserPromise context.commit('updateUser', userInfo) } finally { getUserPromise = null } return context.state.userInfo }, // 登出 async logout(context, payload = {}) { // 是否手動退出 const { manual } = payload if (manual) { await userApi.postLogout() } user.clearToken() context.commit('updateUser', null) }, } }
然后在項目的 main.js
文件中引入
import Vue from 'vue' import App from '@/app.vue' import { router } from '@/router' import store from '@/store/index' const vue = new Vue({ el: '#app', name: 'root', router, store, render: h => h(App), })
封裝的很愉快了,然后就正常操作即可。
this.$store.state.user.isLogined this.$store.state.user.userInfo this.$store.commit('user/updateUser', {}) await this.$store.dispatch('user/logout', { manual: true })
這種通信比較簡單,缺點是數(shù)據(jù)和狀態(tài)比較混亂,不太容易維護。
通過window.localStorage.getItem(key)
獲取數(shù)據(jù)
通過window.localStorage.setItem(key,value)
存儲數(shù)據(jù)
注意用JSON.parse()
/ JSON.stringify()
做數(shù)據(jù)格式轉(zhuǎn)換, localStorage / sessionStorage可以結(jié)合vuex, 實現(xiàn)數(shù)據(jù)的持久保存
,同時使用vuex解決數(shù)據(jù)和狀態(tài)混亂問題.
對于小型的項目,通信十分簡單,這時使用 Vuex 反而會顯得冗余和繁瑣,這種情況最好不要使用 Vuex,可以自己在項目中實現(xiàn)簡單的 Store。
// store.js const store = { debug: true, state: { author: 'yushihu!' }, setAuthorAction (newValue) { if (this.debug) console.log('setAuthorAction triggered with', newValue) this.state.author = newValue }, deleteAuthorAction () { if (this.debug) console.log('deleteAuthorAction triggered') this.state.author = '' } } export default store
上面代碼原理就是,store.js文件
暴露出一個對象 store
,通過引入 store.js 文件
各個頁面來共同維護這個store對象
和 Vuex 一樣,store 中 state 的改變都由 store 內(nèi)部的 action 來觸發(fā),并且能夠通過 console.log()
打印觸發(fā)的痕跡。這種方式十分適合在不需要使用 Vuex 的小項目中應用。
與 $root
訪問根實例的方法相比,這種集中式狀態(tài)管理的方式
能夠在調(diào)試過程中,通過 console.log()
記錄來確定當前變化是如何觸發(fā)的,更容易定位問題。
通過 $root
,任何組件都可以獲取當前組件樹的根 Vue 實例
,通過維護根實例上的 data,就可以實現(xiàn)組件間的數(shù)據(jù)共享
。
//main.js 根實例 new Vue({ el: '#app', store, router, // 根實例的 data 屬性,維護通用的數(shù)據(jù) data: function () { return { author: '' } }, components: { App }, template: '', }); 本文作者{{ $root.author }}
注意:通過這種方式,雖然可以實現(xiàn)通信,但在應用的任何部分,任何時間發(fā)生的任何數(shù)據(jù)變化,都不會留下變更的記錄,這對于稍復雜的應用來說,調(diào)試是致命的,不建議在實際應用中使用。
現(xiàn)在我們來討論一種情況, 我們一開始給出的組件關系圖中A組件與D組件是隔代關系, 那它們之前進行通信有哪些方式呢?
使用props
綁定來進行一級一級的信息傳遞, 如果D組件中狀態(tài)改變需要傳遞數(shù)據(jù)給A, 使用事件系統(tǒng)一級級往上傳遞
使用eventBus
,這種情況下還是比較適合使用, 但是碰到多人合作開發(fā)時, 代碼維護性較低, 可讀性也低
使用Vuex
來進行數(shù)據(jù)管理, 但是如果僅僅是傳遞數(shù)據(jù)
, 而不做中間處理,使用Vuex處理感覺有點大材小用
了.
所以就有了 $attrs
/ $listeners
,通常配合 inheritAttrs
一起使用。
inheritAttrs
默認情況下父作用域的不被認作 props
的 attribute
綁定 (attribute bindings) 將會“回退”且作為普通的 HTML attribute 應用在子組件的根元素上。當撰寫包裹一個目標元素或另一個組件的組件時,這可能不會總是符合預期行為。
通過設置 inheritAttrs
到 false
,這些默認行為將會被去掉。而通過 (同樣是 2.4 新增的) 實例 property $attrs 可以讓這些 attribute 生效,且可以通過 v-bind 顯性的綁定到非根元素上。
注意:這個選項不影響 class
和 style
綁定。
上面是官方描述:還是很難懂。
簡單的說就是
inheritAttrs:true
時繼承除props之外的所有屬性
inheritAttrs:false
只繼承class 和 style屬性
$attrs
:包含了父作用域中不被認為 (且不預期為) props 的特性綁定 (class 和 style 除外
),并且可以通過 v-bind="$attrs"
傳入內(nèi)部組件。當一個組件沒有聲明任何 props
時,它包含所有父作用域的綁定 (class 和 style 除外)。
$listeners
:包含了父作用域中的 (不含 .native 修飾符
) v-on 事件監(jiān)聽器。它可以通過 v-on="$listeners"
傳入內(nèi)部組件。它是一個對象,里面包含了作用在這個組件上的所有事件監(jiān)聽器,相當于子組件繼承了父組件的事件
。
講了這么多文字概念,我們還是來看代碼例子吧:
新建一個 father.vue 組件
child.vue 組件:
grandSon.vue 組件
{{ $attrs }} --- {{ $listeners }}這種方式的傳值雖然說不常用,感覺可讀性不是很好。但其對于組件層級嵌套比較深,使用props會很繁瑣,或者項目比較小,不太適合使用 Vuex 的時候,可以考慮用它
“vue組件間如何進行通信”的內(nèi)容就介紹到這里了,感謝大家的閱讀。如果想了解更多行業(yè)相關的知識可以關注創(chuàng)新互聯(lián)網(wǎng)站,小編將為大家輸出更多高質(zhì)量的實用文章!
當前文章:vue組件間如何進行通信
文章出自:http://weahome.cn/article/giejhp.html