前端面試 vue 部分 (5)——VUE組件之間通訊的方式有哪些

file

VUE組件之間通訊的方式有哪些(SSS)

圖片

常見使用場景能夠分爲三類:javascript

  • 父子通訊:html

    • null
      父向子傳遞數據是經過 props ,子向父是經過 $emit / $on
    • $emit / $bus
    • Vuex
    • 經過父鏈 / 子鏈也能夠通訊( $parent / $children
    • ref 也能夠訪問組件實例
  • 兄弟通訊:vue

    • $emit / $bus
    • Vuex
  • 跨級通訊:java

    • $emit / $bus
    • Vuex
    • provide / inject API
    • $attrs/$listeners

$emit / $busvuex

// main.js 
Vue.prototype.$bus = new Vue() // event Bus 用於無關係組件間的通訊。

A觸發Bapp

//  A 
 this.$bus.$emit('new-messsage-at-me', { 
    data: { conversationID: message.conversationID } 
  })
//  B 
  mounted() { 
    this.$bus.$on('new-messsage-at-me', event => { 
      if ( 
        event.data.conversationID === this.conversation.conversationID && 
        this.conversation.conversationID !== 
          this.currentConversation.conversationID 
      ) { 
        this.hasMessageAtMe = true 
      } 
    }) 
  },

父子組件通訊dom

  1. 父組件向子組件傳值( props ):
//App.vue父組件 
<template> 
  <div id="app"> 
    <users v-bind:users="users"></users>//前者自定義名稱便於子組件調用,後者要傳遞數據名 
  </div> 
</template> 
<script> 
import Users from "./components/Users" 
export default { 
  name: 'App', 
  data(){ 
    return{ 
      users:["Henry","Bucky","Emily"] 
    } 
  }, 
  components:{ 
    "users":Users 
  } 
}
//users子組件  
// 注:組件中的數據共有三種形式:data、props、computed 
<template> 
  <div class="hello"> 
    <ul> 
      <li v-for="user in users">{{user}}</li>//遍歷傳遞過來的值,而後呈現到頁面 
    </ul> 
  </div> 
</template> 
<script> 
export default { 
  name: 'HelloWorld', 
  props:{ 
    users:{           //這個就是父組件中子標籤自定義名字 
      type:Array, 
      required:true 
    } 
  } 
} 
</script>
  1. 子組件向父組件傳值(B 組件中 $emit, A 組件中 v-on ):
// 子組件 
<template> 
  <header> 
    <h1 @click="changeTitle">{{title}}</h1>//綁定一個點擊事件 
  </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:"傳遞的是一個值" 
    } 
  }, 
  methods:{ 
    updateTitle(e){   //聲明這個函數 
      this.title = e; 
    } 
  }, 
  components:{ 
   "app-header":Header, 
  } 
} 
</script>

$ref 與 $parent $childrenide

  • 使用 this.$parent查找當前組件的父組件。
  • 使用 this.$children查找當前組件的直接子組件,能夠遍歷所有子組件, 須要注意 $children 並不保證順序,也不是響應式的。
  • 使用 this.$root查找根組件,並能夠配合$children遍歷所有組件。
  • 使用 this.$refs查找命名子組件( )( this.$refs.one )

$attrs / $listeners函數

  • 二者的出現使得組件之間跨組件的通訊在不依賴 vuex 和事件總線的狀況下變得簡潔,業務清晰。
  • A->B->C 多級組件嵌套須要傳遞數據時,一般使用的方法是經過vuex。若是僅僅是傳遞數據,而不作中間處理,使用 vuex 處理,未免有點殺雞用牛刀。Vue 2.4 版本提供了另外一種方法,使用 v-bind=」$attrs」, 將父組件中不被認爲 props特性綁定的屬性傳入子組件中,一般配合 interitAttrs 選項一塊兒使用。
  • 簡單來講:$attrs 與 $listeners 是兩個「對象」,$attrs 裏存放的是父組件中綁定的非 Props 屬性, 惟一缺點 沒在props定義的屬性 會顯示在生成的html標籤上, 解決辦法:經過inheritAttrs:false,避免頂層容器繼承屬性; $listeners裏存放的是父組件中綁定的非原生事件。

A父組件ui

<template> 
 <div> 
   <child-dom 
    :foo="foo" 
    :coo="coo" 
     v-on:upRocket="reciveRocket" 
   > 
   </child-dom> 
 </div> 
</template> 
<script> 
 import childDom from "@/components/ChildDom.vue"; 
 export default { 
   name:'demoNo', 
   data() { 
     return { 
       foo:"Hello, world", 
        coo:"Hello,rui" 
    } 
  }, 
 components:{childDom}, 
 methods:{ 
   reciveRocket(){ 
      console.log("reciveRocket success") 
   } 
 } 
} 
</script>

B子組件

<template> 
   <div> 
     <p>foo:{{foo}}</p> 
     <p>attrs:{{$attrs}}</p> 
     <childDomChild v-bind="$attrs" v-on="$listeners"></childDomChild> 
   </div> 
</template> 
<script> 
import childDomChild from './childDomChild'; 
export default { 
   name:'child-dom' 
   props:["foo"], 
   inheritAttrs:false, 
} 
</script>

C子組件的子組件

<template>  
   <div> 
       <p>coo:{{coo}}</p> 
       <button @click="startUpRocket">我要發射火箭</button> 
   </div> 
</template> 
<script> 
 export default { 
   name:'childDomChild', 
   props:['coo'], 
   methods:{ 
     startUpRocket(){ 
       this.$emit("upRocket"); 
       console.log("startUpRocket") 
     } 
   } 
 } 
</script>

provide / inject

  • 適用於 隔代組件通訊 祖先組件中經過 provider 來提供變量,而後在子孫組件中經過 inject 來注入變量。 provide / inject API 主要解決了跨級組件間的通訊問題,不過它的使用場景,主要是子組件獲取上級組件的狀態,跨級組件間創建了一種主動提供與依賴注入的關係。
  • 若是是單一的只是拿數據使用,在父組件定義,則在全部子組件都能爲之所用
  • 官網不建議在應用中直接使用該辦法,理由很直接:他怕你"管很差"

1.通常狀況使用都是在app.vue配置爲:

provide () { 
  return { 
    isTest: this 
  } 
},

2.全部子組件均可以引用 拿到app.vue裏面的全部數據

inject: ['isTest'],

歡迎留言~~~

相關文章
相關標籤/搜索