父子組件如何實現通信

概念

組件是 vue.js最強大的功能之一,而組件實例的作用域是相互獨立的,這就意味着不同組件之間的數據無法相互引用。一般來說,組件可以有以下幾種關係:
在這裏插入圖片描述
如上圖所示,A 和 B、B 和 C、B 和 D 都是父子關係,C 和 D 是兄弟關係,A 和 C 是隔代關係(可能隔多代)

針對不同的使用場景,如何選擇行之有效的通信方式?
這也是面試中經常問涉及的一個問題,常見的解決方案有以下幾種:

  • props + $emit (組件封裝用的較多)
  • EventBus (中央事件總線)
  • vuex (用於狀態管理)
  • $attrs / $listeners (組件封裝用的較多)
  • provide / inject (高階組件/組件庫用的較多)
  • $root / $parent / $children / ref(訪問父示例或者子示例,極小情況下會直接修改父組件中的數據)
  • .sync 修飾符 (語法棒棒糖)
  • broadcast/ dispatch (他倆是 [email protected] 中的方法,分別是事件廣播 和 事件派發)

方案一:props / $emit

示例

// 父組件
<template>
  <div id="app">
    <child-component :users="users" @change="change"></child-component> // 與子組件change自定義事件保持一致
           // change($event)接受傳遞過來的文字
    <a @click="add">點擊添加成員</a>
  </div>
</template>

<script>
import child from "./components/Users"
export default {
  name: 'App',
  data(){
    return{
      users:["a","b","c"]
    }
  },
  components:{
    "child-component": child
  },
  methods:{
  	change:function(reslut){
  		console.log(reslut)
  	}
  }
}

// child子組件
<template>
  <div class="hello">
    <ul>
      <li v-for="item in userList">{{item}}</li>
    </ul>
  </div>
</template>

<script>
export default {
  name: 'child',
  props:{
    users:{
      type:Array,
      required:true
    }
  },
  data:function(){
     userList: users
   },
  methods:{
  	add:function(){
       userList.pusth('d');
       this.$emit('change', userList); //自定義事件  傳遞值“子向父組件傳值”
     }
  }
}
</script>

說明

父組件通過props向下傳遞數據給子組件,子組件通過events給父組件發送消息,實際上就是子組件把自己的數據發送到父組件

方案二:中央事件總線

介紹

通過一個空的Vue實例作爲中央事件總線(事件中心),用它來觸發事件和監聽事件,巧妙而輕量地實現了任何組件間的通信,包括父子、兄弟、跨級。當我們的項目比較大時,可以選擇更好的狀態管理解決方案vuex

方法

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

示例

<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">將數據發送給C組件</button>
  </div>
</template>

<template id="b">
  <div>
    <h3>B組件:{{age}}</h3>
    <button @click="send">將數組發送給C組件</button>
  </div>
</template>

<template id="c">
  <div>
    <h3>C組件:{{name}}{{age}}</h3>
  </div>
</template>

<script>
var Event = new Vue(); //定義一個空的Vue實例
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() {//在模板編譯完成後執行
	 Event.$on('data-a',name => {
	     this.name = name;//箭頭函數內部不會產生新的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

在這裏插入圖片描述

原理

Vuex原理請閱讀《Vuex是什麼?

各個模塊的作用

  • Vue ComponentsVue組件。HTML頁面上,負責接收用戶操作等交互行爲,執行dispatch方法觸發對應action進行迴應。
  • dispatch:操作行爲觸發方法,是唯一能執行action的方法。
  • actions:操作行爲處理模塊,由組件中的$store.dispatch('action 名稱', data1)來觸發。然後由commit()來觸發mutation的調用 , 間接更新 state。負責處理Vue Components接收到的所有交互行爲。包含同步/異步操作,支持多個同名方法,按照註冊的順序依次觸發。向後臺API請求的操作就在這個模塊中進行,包括觸發其他action以及提交mutation的操作。該模塊提供了Promise的封裝,以支持action的鏈式觸發。
  • commit:狀態改變提交操作方法。對·mutation·進行提交,是唯一能執行·mutation·的方法。
  • mutations:狀態改變操作方法,由actions中的commit('mutation 名稱')來觸發。是Vuex修改state的唯一推薦方法。該方法只能進行同步操作,且方法名只能全局唯一。操作之中會有一些hook暴露出來,以進行state的監控等。
  • state:頁面狀態管理容器對象。集中存儲Vue componentsdata對象的零散數據,全局唯一,以進行統一的狀態管理。頁面顯示所需的數據從該對象中進行讀取,利用Vue的細粒度數據響應機制來進行高效的狀態更新。
  • gettersstate對象讀取方法。圖中沒有單獨列出該模塊,應該被包含在了render中,Vue Components通過該方法讀取全局state對象。

方案四:$attrs / $listeners

介紹

多級組件嵌套需要傳遞數據時,通常使用的方法是通過vuex。但如果僅僅是傳遞數據,而不做中間處理,使用 vuex處理,未免有點大材小用。爲此Vue2.4 版本提供了另一種方法:$attrs / $listeners

  • $attrs:包含了父作用域中不被 prop 所識別 (且獲取) 的特性綁定 (class 和 style 除外)。當一個組件沒有聲明任何 prop時,這裏會包含所有父作用域的綁定 (class 和 style 除外),並且可以通過 v-bind="$attrs" 傳入內部組件。通常配合 inheritAttrs選項一起使用。
  • $listeners:包含了父作用域中的 (不含 .native 修飾器的) v-on 事件監聽器。它可以通過 v-on="$listeners" 傳入內部組件

示例

// index.vue
<template>
  <div>
    <child1
      :name="name"
      :sex="sex"
      :age="age"
    ></child1>
  </div>
</template>

<script>
const child1 = () => import("./child1.vue");
export default {
  components: { child1 },
  data() {
    return {
      name: "baby",
      bsexo: "female",
      age: 20
    };
  }
};
</script>


// child1.vue
<template>
  <div>
    <p>name: {{ name}}</p>
    <p>child1 的 $attrs: {{ $attrs }}</p>
    <child2 v-bind="$attrs"></child2>
  </div>
</template>

<script>
const child2 = () => import("./child2.vue");
export default {
  components: {
    child2
  },
  inheritAttrs: false, // 可以關閉自動掛載到組件根元素上的沒有在props聲明的屬性
  props: {
    foo: String // foo作爲props屬性綁定
  },
  created() {
    console.log(this.$attrs);
  }
};
</script>


// child2.vue
<template>
  <div>
    <p>name: {{ name}}</p>
    <p>chil2: {{ $attrs }}</p>
    <child3 v-bind="$attrs"></child3>
  </div>
</template>
<script>
const child3 = () => import("./child3.vue");
export default {
  components: {
    child3
  },
  inheritAttrs: false,
  props: {
    boo: String
  },
  created() {
    console.log(this.$attrs);
  }
};
</script>


// child3.vue
<template>
  <div>
    <p>child3: {{ $attrs }}</p>
  </div>
</template>
<script>
export default {
  props: {
    coo: String,
    title: String
  }
};
</script>

說明

$attrs表示沒有繼承數據的對象,格式爲{屬性名:屬性值}。Vue2.4提供了$attrs , $listeners 來傳遞數據與事件,跨級組件之間的通訊變得更簡單。
$attrs$listeners是兩個對象,$attrs 裏存放的是父組件中綁定的非 Props 屬性,$listeners裏存放的是父組件中綁定的非原生事件。

方案五:provide / inject

簡介

Vue2.2.0新增API,這對選項需要一起使用,以允許一個祖先組件向其所有子孫後代注入一個依賴,不論組件層次有多深,並在起上下游關係成立的時間裏始終生效。一言而蔽之:祖先組件中通過provider來提供變量,然後在子孫組件中通過inject來注入變量。
provide / inject API 主要解決了跨級組件間的通信問題,不過它的使用場景,主要是子組件獲取上級組件的狀態,跨級組件間建立了一種主動提供與依賴注入的關係。

語法

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

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

需要注意的是:provideinject綁定並不是可響應的。這是刻意爲之的。然而,如果你傳入了一個可監聽的對象,那麼其對象的屬性還是可響應的----vue官方文檔
所以,上面 A.vue 的 name如果改變了,B.vue 的 ``this.name `是不會改變的,仍然是 baby。

怎麼實現數據響應式

一般來說,有兩種辦法:

  • provide祖先組件的實例,然後在子孫組件中注入依賴,這樣就可以在子孫組件中直接修改祖先組件的實例的屬性,不過這種方法有個缺點就是這個實例上掛載很多沒有必要的東西比如propsmethods
  • 使用2.6最新API Vue.observable 優化響應式 provide(推薦)

我們來看個例子:孫組件D、E和F獲取A組件傳遞過來的color值,並能實現數據響應式變化,即A組件的color變化後,組件D、E、F會跟着變(核心代碼如下:)
在這裏插入圖片描述

// A 組件 
<div>
      <h1>A 組件</h1>
      <button @click="() => changeColor()">改變color</button>
      <ChildrenB />
      <ChildrenC />
</div>
......
  data() {
    return {
      color: "blue"
    };
  },
  // provide() {
  //   return {
  //     theme: {
  //       color: this.color //這種方式綁定的數據並不是可響應的
  //     } // 即A組件的color變化後,組件D、E、F不會跟着變
  //   };
  // },
  provide() {
    return {
      theme: this//方法一:提供祖先組件的實例
    };
  },
  methods: {
    changeColor(color) {
      if (color) {
        this.color = color;
      } else {
        this.color = this.color === "blue" ? "red" : "blue";
      }
    }
  }
  // 方法二:使用2.6最新API Vue.observable 優化響應式 provide
  // provide() {
  //   this.theme = Vue.observable({
  //     color: "blue"
  //   });
  //   return {
  //     theme: this.theme
  //   };
  // },
  // methods: {
  //   changeColor(color) {
  //     if (color) {
  //       this.theme.color = color;
  //     } else {
  //       this.theme.color = this.theme.color === "blue" ? "red" : "blue";
  //     }
  //   }
  // }

// F 組件 
<template functional>
  <div class="border2">
    <h3 :style="{ color: injections.theme.color }">F 組件</h3>
  </div>
</template>
<script>
export default {
  inject: {
    theme: {
      //函數式組件取值不一樣
      default: () => ({})
    }
  }
};
</script>

雖說provideinject主要爲高階插件/組件庫提供用例,但如果你能在業務中熟練運用,可以達到事半功倍的效果

方案六:$parent / $childrenref

簡介

  • ref:如果在普通的 DOM 元素上使用,引用指向的就是 DOM 元素;如果用在子組件上,引用就指向組件實例
  • $parent / $children:訪問父 / 子實例
    需要注意的是:這兩種都是直接得到組件實例,使用後可以直接調用組件的方法或訪問數據
// child-conpoent 子組件
export default {
  data () {
    return {
      title: 'Vue.js'
    }
  },
  methods: {
    sayHello () {
      window.alert('Hello');
    }
  }
}

// 父組件
<template>
  <child-conpoent ref="child"></child-conpoent>
</template>

<script>
  export default {
    mounted () {
      const comA = this.$refs.child;
      console.log(child.title);  // Vue.js
      comA.sayHello();  // 彈窗
    }
  }
</script>

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

方案七: .sync 修飾符

簡介

這個傢伙在 [email protected] 的時候曾作爲雙向綁定功能存在,即子組件可以修改父組件中的值。因爲它違反了單向數據流的設計理念,所以在 [email protected] 的時候被幹掉了。但是在 [email protected]+ 以上版本又重新引入了這個 .sync 修飾符。但是這次它只是作爲一個編譯時的語法糖存在。它會被擴展爲一個自動更新父組件屬性的 v-on 監聽器。說白了就是讓我們手動進行更新父組件中的值了,從而使數據改動來源更加的明顯。下面引入自官方的一段話:

在有些情況下,我們可能需要對一個 prop 進行“雙向綁定”。不幸的是,真正的雙向綁定會帶來維護上的問題,因爲子組件可以修改父組件,且在父組件和子組件都沒有明顯的改動來源。

<text-document
  v-bind:title="doc.title"
  v-on:update:title="doc.title = $event">
</text-document>

於是我們可以用 .sync 語法糖簡寫成如下形式:

<text-document v-bind:title.sync="doc.title"></text-document>

示例

<div id="app">
  <login :name.sync="userName"></login> {{ userName }}
</div>

let Login = Vue.extend({
  template: `
    <div class="input-group">
      <label>姓名:</label>
      <input v-model="text">
    </div>
  `,
  props: ['name'],
  data () {
    return {
      text: ''
    }
  },
  watch: {
    text (newVal) {
      this.$emit('update:name', newVal)
    }
  }
})

new Vue({
  el: '#app',
  data: {
    userName: ''
  },
  components: {
    Login
  }
})

官方語法是:update:myPropName 其中 myPropName 表示要更新的 prop 值。當然如果你不用 .sync語法糖使用上面的 .$emit也能達到同樣的效果。

方案八:broadcast / dispatch

他倆是 [email protected] 中的方法,分別是事件廣播 和 事件派發。雖然 [email protected] 裏面刪掉了,但可以模擬這兩個方法。可以借鑑 Element實現。有時候還是非常有用的,比如我們在開發樹形組件的時候等等。

參考資料:
https://cn.vuejs.org/v2/guide/components-props.html
https://juejin.im/post/5cde0b43f265da03867e78d3

發表評論
所有評論
還沒有人評論,想成為第一個評論的人麼? 請在上方評論欄輸入並且點擊發布.
相關文章