欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

vue組件間通信六種方式(總結(jié)篇)

 更新時(shí)間:2019年05月15日 08:44:01   作者:浪里行舟  
組件是 vue.js最強(qiáng)大的功能之一,而組件實(shí)例的作用域是相互獨(dú)立的,這就意味著不同組件之間的數(shù)據(jù)無法相互引用。這篇文章主要介紹了vue組件間通信六種方式,需要的朋友可以參考下

前言

組件是 vue.js最強(qiáng)大的功能之一,而組件實(shí)例的作用域是相互獨(dú)立的,這就意味著不同組件之間的數(shù)據(jù)無法相互引用。一般來說,組件可以有以下幾種關(guān)系:

如上圖所示,A 和 B、B 和 C、B 和 D 都是父子關(guān)系,C 和 D 是兄弟關(guān)系,A 和 C 是隔代關(guān)系(可能隔多代)。

針對不同的使用場景,如何選擇行之有效的通信方式?這是我們所要探討的主題。本文總結(jié)了vue組件間通信的幾種方式,如props、 $emit / $on 、vuex、 $parent / $children 、 $attrs / $listeners 和provide/inject,以通俗易懂的實(shí)例講述這其中的差別及使用場景,希望對小伙伴有些許幫助。

本文的代碼請猛戳github博客 ,紙上得來終覺淺,大家動(dòng)手多敲敲代碼!

方法一、 props / $emit

父組件A通過props的方式向子組件B傳遞,B to A 通過在 B 組件中 $emit, A 組件中 v-on 的方式實(shí)現(xiàn)。

1.父組件向子組件傳值

接下來我們通過一個(gè)例子,說明父組件如何向子組件傳遞值:在子組件Users.vue中如何獲取父組件App.vue中的數(shù)據(jù)  users:["Henry","Bucky","Emily"]

//App.vue父組件 
<template> 
 <div id="app"> 
 <users v-bind:users="users"></users>//前者自定義名稱便于子組件調(diào)用,后者要傳遞數(shù)據(jù)名 
 </div> 
</template> 
<script> 
import Users from "./components/Users" 
export default { 
 name: 'App', 
 data(){ 
 return{ 
 users:["Henry","Bucky","Emily"] 
 } 
 }, 
 components:{ 
 "users":Users 
 } 
} 
//users子組件 
<template> 
 <div class="hello"> 
 <ul> 
 <li v-for="user in users">{{user}}</li>//遍歷傳遞過來的值,然后呈現(xiàn)到頁面 
 </ul> 
 </div> 
</template> 
<script> 
export default { 
 name: 'HelloWorld', 
 props:{ 
 users:{ //這個(gè)就是父組件中子標(biāo)簽自定義名字 
 type:Array, 
 required:true 
 } 
 } 
} 
</script>

總結(jié):父組件通過props向下傳遞數(shù)據(jù)給子組件。注:組件中的數(shù)據(jù)共有三種形式:data、props、computed

2.子組件向父組件傳值(通過事件形式)

接下來我們通過一個(gè)例子,說明子組件如何向父組件傳遞值:當(dāng)我們點(diǎn)擊“Vue.js Demo”后,子組件向父組件傳遞值,文字由原來的“傳遞的是一個(gè)值”變成“子向父組件傳值”,實(shí)現(xiàn)子組件向父組件值的傳遞。

// 子組件 
<template> 
 <header> 
 <h1 @click="changeTitle">{{title}}</h1>//綁定一個(gè)點(diǎn)擊事件 
 </header> 
</template> 
<script> 
export default { 
 name: 'app-header', 
 data() { 
 return { 
 title:"Vue.js Demo" 
 } 
 }, 
 methods:{ 
 changeTitle() { 
 this.$emit("titleChanged","子向父組件傳值");//自定義事件 傳遞值“子向父組件傳值” 
 } 
 } 
} 
</script> 
// 父組件 
<template> 
 <div id="app"> 
 <app-header v-on:titleChanged="updateTitle" ></app-header>//與子組件titleChanged自定義事件保持一致 
 // updateTitle($event)接受傳遞過來的文字 
 <h2>{{title}}</h2> 
 </div> 
</template> 
<script> 
import Header from "./components/Header" 
export default { 
 name: 'App', 
 data(){ 
 return{ 
 title:"傳遞的是一個(gè)值" 
 } 
 }, 
 methods:{ 
 updateTitle(e){ //聲明這個(gè)函數(shù) 
 this.title = e; 
 } 
 }, 
 components:{ 
 "app-header":Header, 
 } 
} 
</script>

總結(jié):子組件通過events給父組件發(fā)送消息,實(shí)際上就是子組件把自己的數(shù)據(jù)發(fā)送到父組件。

方法二、 $emit / $on

這種方法通過一個(gè)空的Vue實(shí)例作為中央事件總線(事件中心),用它來觸發(fā)事件和監(jiān)聽事件,巧妙而輕量地實(shí)現(xiàn)了任何組件間的通信,包括父子、兄弟、跨級。當(dāng)我們的項(xiàng)目比較大時(shí),可以選擇更好的狀態(tài)管理解決方案vuex 。

1.具體實(shí)現(xiàn)方式:  

var Event=new Vue(); 
Event.$emit(事件名,數(shù)據(jù)); 
Event.$on(事件名,data => {});

2.舉個(gè)例子:兄弟組件有三個(gè),分別是A、B、C,C組件如何獲取A或者B組件的數(shù)據(jù)

 

<div id="itany"> 
 <my-a></my-a> 
 <my-b></my-b> 
 <my-c></my-c> 
</div> 
<template id="a"> 
 <div> 
 <h3>A組件:{{name}}</h3> 
 <button @click="send">將數(shù)據(jù)發(fā)送給C組件</button> 
 </div> 
</template> 
<template id="b"> 
 <div> 
 <h3>B組件:{{age}}</h3> 
 <button @click="send">將數(shù)組發(fā)送給C組件</button> 
 </div> 
</template> 
<template id="c"> 
 <div> 
 <h3>C組件:{{name}},{{age}}</h3> 
 </div> 
</template> 
<script> 
var Event = new Vue();//定義一個(gè)空的Vue實(shí)例 
var A = { 
 template: '#a', 
 data() { 
 return { 
 name: 'tom' 
 } 
 }, 
 methods: { 
 send() { 
 Event.$emit('data-a', this.name); 
 } 
 } 
} 
var B = { 
 template: '#b', 
 data() { 
 return { 
 age: 20 
 } 
 }, 
 methods: { 
 send() { 
 Event.$emit('data-b', this.age); 
 } 
 } 
} 
var C = { 
 template: '#c', 
 data() { 
 return { 
 name: '', 
 age: "" 
 } 
 }, 
 mounted() {//在模板編譯完成后執(zhí)行 
 Event.$on('data-a',name => { 
 this.name = name;//箭頭函數(shù)內(nèi)部不會(huì)產(chǎn)生新的this,這邊如果不用=>,this指代Event 
 }) 
 Event.$on('data-b',age => { 
 this.age = age; 
 }) 
 } 
} 
var vm = new Vue({ 
 el: '#itany', 
 components: { 
 'my-a': A, 
 'my-b': B, 
 'my-c': C 
 } 
}); 
</script> 

方法三、vuex

1.簡要介紹Vuex原理

Vuex實(shí)現(xiàn)了一個(gè)單向數(shù)據(jù)流,在全局擁有一個(gè)State存放數(shù)據(jù),當(dāng)組件要更改State中的數(shù)據(jù)時(shí),必須通過Mutation進(jìn)行,Mutation同時(shí)提供了訂閱者模式供外部插件調(diào)用獲取State數(shù)據(jù)的更新。而當(dāng)所有異步操作(常見于調(diào)用后端接口異步獲取更新數(shù)據(jù))或批量的同步操作需要走Action,但Action也是無法直接修改State的,還是需要通過Mutation來修改State的數(shù)據(jù)。最后,根據(jù)State的變化,渲染到視圖上。

2.簡要介紹各模塊在流程中的功能:

Vue Components:Vue組件。HTML頁面上,負(fù)責(zé)接收用戶操作等交互行為,執(zhí)行dispatch方法觸發(fā)對應(yīng)action進(jìn)行回應(yīng)。

dispatch:操作行為觸發(fā)方法,是唯一能執(zhí)行action的方法。

actions: 操作行為處理模塊,由組件中的 $store.dispatch('action 名稱', data1) 來觸發(fā)。然后由commit()來觸發(fā)mutation的調(diào)用 , 間接更新 state 。負(fù)責(zé)處理Vue Components接收到的所有交互行為。包含同步/異步操作,支持多個(gè)同名方法,按照注冊的順序依次觸發(fā)。向后臺(tái)API請求的操作就在這個(gè)模塊中進(jìn)行,包括觸發(fā)其他action以及提交mutation的操作。該模塊提供了Promise的封裝,以支持action的鏈?zhǔn)接|發(fā)。

commit:狀態(tài)改變提交操作方法。對mutation進(jìn)行提交,是唯一能執(zhí)行mutation的方法。

mutations: 狀態(tài)改變操作方法,由actions中的 commit('mutation 名稱') 來觸發(fā) 。是Vuex修改state的唯一推薦方法。該方法只能進(jìn)行同步操作,且方法名只能全局唯一。操作之中會(huì)有一些hook暴露出來,以進(jìn)行state的監(jiān)控等。

state:頁面狀態(tài)管理容器對象。集中存儲(chǔ)Vue components中data對象的零散數(shù)據(jù),全局唯一,以進(jìn)行統(tǒng)一的狀態(tài)管理。頁面顯示所需的數(shù)據(jù)從該對象中進(jìn)行讀取,利用Vue的細(xì)粒度數(shù)據(jù)響應(yīng)機(jī)制來進(jìn)行高效的狀態(tài)更新。

getters:state對象讀取方法。圖中沒有單獨(dú)列出該模塊,應(yīng)該被包含在了render中,Vue Components通過該方法讀取全局state對象。

3.Vuex與localStorage

vuex 是 vue 的狀態(tài)管理器,存儲(chǔ)的數(shù)據(jù)是響應(yīng)式的。但是并不會(huì)保存起來,刷新之后就回到了初始狀態(tài), 具體做法應(yīng)該在vuex里數(shù)據(jù)改變的時(shí)候把數(shù)據(jù)拷貝一份保存到localStorage里面,刷新之后,如果localStorage里有保存的數(shù)據(jù),取出來再替換store里的state。

let defaultCity = "上海" 
try { // 用戶關(guān)閉了本地存儲(chǔ)功能,此時(shí)在外層加個(gè)try...catch 
 if (!defaultCity){ 
 defaultCity = JSON.parse(window.localStorage.getItem('defaultCity')) 
 } 
}catch(e){} 
export default new Vuex.Store({ 
 state: { 
 city: defaultCity 
 }, 
 mutations: { 
 changeCity(state, city) { 
 state.city = city 
 try { 
 window.localStorage.setItem('defaultCity', JSON.stringify(state.city)); 
 // 數(shù)據(jù)改變的時(shí)候把數(shù)據(jù)拷貝一份保存到localStorage里面 
 } catch (e) {} 
 } 
 } 
})

這里需要注意的是:由于vuex里,我們保存的狀態(tài),都是數(shù)組,而localStorage只支持字符串,所以需要用JSON轉(zhuǎn)換:

JSON.stringify(state.subscribeList); // array -> string 
JSON.parse(window.localStorage.getItem("subscribeList")); // string -> array 

方法四、 $attrs / $listeners

1.簡介

多級組件嵌套需要傳遞數(shù)據(jù)時(shí),通常使用的方法是通過vuex。但如果僅僅是傳遞數(shù)據(jù),而不做中間處理,使用 vuex 處理,未免有點(diǎn)大材小用。為此Vue2.4 版本提供了另一種方法, 當(dāng)一個(gè)組件沒有聲明任何 prop 時(shí),這里會(huì)包含所有父作用域的綁定 (class 和 style 除外),并且可以通過 v-bind="$attrs" 傳入內(nèi)部組件。通常配合 interitAttrs 選項(xiàng)一起使用 。

// demo.vue 
 <template> 
 <div> 
 <child-com:foo="foo":boo="boo":coo="coo":doo="doo"></child-com> 
 </div> 
 </tempalte> 
 <script> 
 const childCom = ()=> import('./childCom1.vue') 
 export default { 
 data () { 
 return { 
 foo: 'Hello World!', 
 boo: 'Hello Javascript!', 
 coo: 'Hello Vue', 
 doo: 'Last' 
 } 
 }, 
 components: { childCom } 
 } 
 </script> 
// childCom1.vue 
<template> 
 <div> 
 <p>foo: {{ foo }}</p> 
 <p>attrs: {{ $attrs }}</p> 
 <child-com2 v-bind="$attrs"></child-com2> 
 </div> 
</template> 
<script> 
const childCom2 = ()=> import('./childCom2.vue') 
export default { 
 props: ['foo'], // foo作為props屬性綁定 
 inheritAttrs: false, 
 created () { 
 console.log(this.$attrs) // { boo: 'Hello Javascript!', coo: 'Hello Vue', doo: 'Last' } 
 } 
} 
</script> 
// childCom2.vue 
<template> 
 <div> 
 <p>boo: {{ boo }}</p> 
 <p>attrs: {{ $attrs }}</p> 
 <child-com3 v-bind="$attrs"></child-com3> 
 </div> 
</template> 
<script> 
const childCom3 = ()=> import('./childCom3.vue') 
export default { 
 props: ['boo'] // boo作為props屬性綁定 
 inheritAttrs: false, 
 created () { 
 console.log(this.$attrs) // { coo: 'Hello Vue', doo: 'Last' } 
 } 
} 
</script>

$attrs 表示沒有繼承數(shù)據(jù)的對象,格式為{屬性名:屬性值}。Vue2.4提供了 $attrs  ,  $listeners  來傳遞數(shù)據(jù)與事件,跨級組件之間的通訊變得更簡單

方法五、provide/inject

1.簡介

Vue2.2.0新增API,這對選項(xiàng)需要一起使用, 以允許一個(gè)祖先組件向其所有子孫后代注入一個(gè)依賴,不論組件層次有多深,并在起上下游關(guān)系成立的時(shí)間里始終生效 。一言而蔽之: 祖先組件中通過provider來提供變量,然后在子孫組件中通過inject來注入變量。

2.舉個(gè)例子

假設(shè)有兩個(gè)組件: A.vue 和 B.vue,B 是 A 的子組件

// A.vue 
export default { 
 provide: { 
 name: '浪里行舟' 
 } 
} 
// B.vue 
export default { 
 inject: ['name'], 
 mounted () { 
 console.log(this.name); // 浪里行舟 
 } 
}

可以看到,在 A.vue 里,我們設(shè)置了一個(gè)  provide: name ,值為 浪里行舟,它的作用就是將  name  這個(gè)變量提供給它的所有子組件。而在 B.vue 中,通過  inject  注入了從 A 組件中提供的  name  變量,那么在組件 B 中,就可以直接通過  this.name  訪問這個(gè)變量了,它的值也是 浪里行舟。這就是 provide / inject API 最核心的用法。

需要注意的是: provide 和 inject 綁定并不是可響應(yīng)的 。不過,如果你傳入了一個(gè)可監(jiān)聽的對象,那么其對象的屬性還是可響應(yīng)的。   所以,上面 A.vue 的 name 如果改變了,B.vue 的 this.name 是不會(huì)改變的,仍然是 浪里行舟。

方法六、 $parent  /  $children 與  ref

ref :如果在普通的 DOM 元素上使用,引用指向的就是 DOM 元素;如果用在子組件上,引用就指向組件實(shí)例

$parent  /  $children :訪問父 / 子實(shí)例

需要注意的是:這兩種都是直接得到組件實(shí)例,使用后可以直接調(diào)用組件的方法或訪問數(shù)據(jù)。我們先來看個(gè)用  ref 來訪問組件的例子:

// component-a 子組件 
export default { 
 data () { 
 return { 
 title: 'Vue.js' 
 } 
 }, 
 methods: { 
 sayHello () { 
 window.alert('Hello'); 
 } 
 } 
} 
// 父組件 
<template> 
 <component-a ref="comA"></component-a> 
</template> 
<script> 
 export default { 
 mounted () { 
 const comA = this.$refs.comA; 
 console.log(comA.title); // Vue.js 
 comA.sayHello(); // 彈窗 
 } 
 } 
</script>

不過, 這兩種方法的弊端是,無法在跨級或兄弟間通信 。

// parent.vue 
<component-a></component-a> 
<component-b></component-b> 
<component-b></component-b>

我們想在 component-a 中,訪問到引用它的頁面中(這里就是 parent.vue)的兩個(gè) component-b 組件,那這種情況下,就得配置額外的插件或工具了,比如 Vuex 和 Bus 的解決方案。

總結(jié)

常見使用場景可以分為三類:

  • 父子通信:   父向子傳遞數(shù)據(jù)是通過 props,子向父是通過 events( $emit );通過父鏈 / 子鏈也可以通信( $parent  /  $children );ref 也可以訪問組件實(shí)例;provide / inject API。
  • 兄弟通信:   Bus;Vuex;
  • 跨級通信:   Bus;Vuex;provide / inject API、 $attrs/$listeners

以上所述是小編給大家介紹的vue組件間通信六種方式(總結(jié)篇),希望對大家有所幫助,如果大家有任何疑問請給我留言,小編會(huì)及時(shí)回復(fù)大家的。在此也非常感謝大家對腳本之家網(wǎng)站的支持!
如果你覺得本文對你有幫助,歡迎轉(zhuǎn)載,煩請注明出處,謝謝!

相關(guān)文章

  • vue中的slot封裝組件彈窗

    vue中的slot封裝組件彈窗

    這篇文章主要介紹了vue中的slot封裝組件彈窗,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-05-05
  • vue axios用法教程詳解

    vue axios用法教程詳解

    axios是vue-resource后出現(xiàn)的Vue請求數(shù)據(jù)的插件。下面我們通過本文給大家介紹vue axios用法教程詳解,感興趣的朋友一起看看吧
    2017-07-07
  • Vue.js 使用v-cloak后仍顯示變量的解決方法

    Vue.js 使用v-cloak后仍顯示變量的解決方法

    這篇文章主要介紹了Vue.js 使用v-cloak后仍顯示變量的解決方法 ,文中給大家提到了v-cloak的用法,需要的朋友可以參考下
    2018-11-11
  • vue-cli3項(xiàng)目打包后自動(dòng)化部署到服務(wù)器的方法

    vue-cli3項(xiàng)目打包后自動(dòng)化部署到服務(wù)器的方法

    這篇文章主要介紹了vue-cli3項(xiàng)目打包后自動(dòng)化部署到服務(wù)器的方法,本文給大家介紹的非常詳細(xì),對大家的學(xué)習(xí)或工作具有一定的參考借鑒價(jià)值,需要的朋友可以參考下
    2020-09-09
  • VUE中的自定義指令鉤子函數(shù)講解

    VUE中的自定義指令鉤子函數(shù)講解

    這篇文章主要介紹了VUE中的自定義指令鉤子函數(shù),具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-08-08
  • Vue CLI 3搭建vue+vuex最全分析(推薦)

    Vue CLI 3搭建vue+vuex最全分析(推薦)

    這篇文章主要介紹了Vue CLI 3搭建vue+vuex最全分析(推薦),小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2018-09-09
  • vue3?錨點(diǎn)定位的多種實(shí)現(xiàn)方式

    vue3?錨點(diǎn)定位的多種實(shí)現(xiàn)方式

    這篇文章主要介紹了vue3?多種方法的錨點(diǎn)定位,使用?Vue?Router?導(dǎo)航守衛(wèi)可以簡化導(dǎo)航邏輯、統(tǒng)一管理導(dǎo)航邏輯和進(jìn)行權(quán)限控制,但需要學(xué)習(xí)和理解相關(guān)概念,并且需要手動(dòng)編寫和管理導(dǎo)航守衛(wèi)的邏輯,本文給大家介紹的非常詳細(xì),需要的朋友可以參考下
    2023-07-07
  • vue2如何實(shí)現(xiàn)vue3的teleport

    vue2如何實(shí)現(xiàn)vue3的teleport

    這篇文章主要介紹了vue2如何實(shí)現(xiàn)vue3的teleport,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-04-04
  • 使用Vite構(gòu)建Vue3項(xiàng)目的流程步驟

    使用Vite構(gòu)建Vue3項(xiàng)目的流程步驟

    在現(xiàn)代前端開發(fā)的世界中,Vue 3 已然成為了一個(gè)備受喜愛的框架,而 Vite 作為一個(gè)新興的構(gòu)建工具,以其極高的效率和簡捷的配置方式席卷了開發(fā)者圈,本文我們將一步一步展示如何使用 Vite 構(gòu)建一個(gè) Vue 3 項(xiàng)目,從創(chuàng)建項(xiàng)目到最后的構(gòu)建階段,需要的朋友可以參考下
    2024-07-07
  • vue 虛擬dom的patch源碼分析

    vue 虛擬dom的patch源碼分析

    這篇文章主要介紹了vue 虛擬dom的patch源碼分析,小編覺得挺不錯(cuò)的,現(xiàn)在分享給大家,也給大家做個(gè)參考。一起跟隨小編過來看看吧
    2018-03-03

最新評論