溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

vue如何實現(xiàn)組件通信

發(fā)布時間:2022-03-03 13:42:54 來源:億速云 閱讀:146 作者:小新 欄目:開發(fā)技術

這篇文章主要介紹vue如何實現(xiàn)組件通信,文中介紹的非常詳細,具有一定的參考價值,感興趣的小伙伴們一定要看完!

前言

在vue中, 組件的關系不外乎以下三種:

vue如何實現(xiàn)組件通信

組件是需要通信的,在開發(fā)中,常用到的通信方式有:vuex、eventBus、以及props與emit、$parent與$children,除此之外,還有provide與inject、$attrs與$listeners等。

一、vuex

這個相信大家用的很多了,簡單回顧一下:

  • State:放狀態(tài)的地方

  • Mutation:唯一修改狀態(tài)的地方,不支持異步

  • Action:通過調(diào)用Mutation中的方法來達到修改狀態(tài)的目的,支持異步

  • Getter:可以理解為計算屬性

  • Module:模塊,每個模塊擁有自己的 state、mutation、action、getter

簡單的使用這里不贅述,提一下module里面的命名空間。

如果希望你的模塊具有更高的封裝度和復用性,你可以通過添加 namespaced: true 的方式使其成為帶命名空間的模塊。當模塊被注冊后,它的所有 getter、action 及 mutation 都會自動根據(jù)模塊注冊的路徑調(diào)整命名

vue如何實現(xiàn)組件通信

這樣,在使用的時候我們就可以這樣用了:

vue如何實現(xiàn)組件通信

二、eventBus

這個稱為‘事件總線’,簡單看下是怎么使用的:

  • 初始化

首先是初始化一個eventBus,可以綁定到vue原型上,也可以綁定到window對象上,還可以抽出來當做一個模塊,在需要的時候再引入。這里直接綁定到vue原型上:

vue如何實現(xiàn)組件通信

  • 創(chuàng)建事件和刪除事件

在需要的組件上創(chuàng)建和刪除事件:

vue如何實現(xiàn)組件通信

  • 觸發(fā)事件

最后就是在需要的地方觸發(fā)事件了

vue如何實現(xiàn)組件通信

三、props/emit

這個不用多說了,父子通信用的最多的應該就是這個了。當然,如果以子組件為跳板,也可以做到祖孫之間通信,不過比較麻煩。不建議這樣操作。

四、$parent/$children

$parent直接訪問的就是父實例,而$children則返回的是實例數(shù)組。所以我一般都是$parent搭配$refs使用。

五、$attrs/$listeners

這兩個可能會用的比較少,來看下官網(wǎng)的介紹:

vue如何實現(xiàn)組件通信

怎么理解呢,簡單來講就是,$attrs接收除了prop、style、class之外的所有綁定屬性,$listeners則接收除了被.native修飾的所有綁定事件。具體來看下例子:

<template>
  <div>
    <p>父組件</p>
    <input type="text" v-model="formData.inputValue" />
    <p>子組件</p>
    <Son
      :inputValue="formData.inputValue"
      :otherValue="otherValue"
      @success="success"
      @input.native="handleInput"
      v-bind="$attrs"
      v-on="$listeners"
    ></Son>
  </div>
</template>
<script>
import Son from "./son.vue";
export default {
  components: { Son },
  provide() {
    return {
      father: this.formData,
    };
  },
  data() {
    return {
      formData: {
        inputValue: "123",
      },
      otherValue: 999,
    };
  },
  methods: {
    success(data) {
      console.log(data);
    },
    handleInput() {},
  },
};
</script>
<template>
  <div>
    <input type="text" v-model="inputValue" @change="handleChange" />
  </div>
</template>
<script>
export default {
  props: {
    inputValue: String,
  },
  created() {
    console.log(this.$attrs, "son---$attrs");
    console.log(this.$listeners, "son---$listeners");
  },
  methods: {
    handleChange() {
      this.father.inputValue = this.inputValue;
    },
  },
};
</script>

按照之前的理解,$attrs應該只能接收到otherValue,$listeners則只能接收到success事件,看下打印結果:

vue如何實現(xiàn)組件通信

結果確實也是這樣的。除此之外,還可傳遞給孫組件:

<template>
  <div>
    <input type="text" v-model="inputValue" @change="handleChange" />
    <GrandSon v-bind="$attrs" v-on="$listeners"></GrandSon>
  </div>
</template>
<script>
import GrandSon from "./grandSon.vue";
export default {
  components: { GrandSon },
  props: {
    inputValue: String,
  },
  created() {
    console.log(this.$attrs, "son---$attrs");
    console.log(this.$listeners, "son---$listeners");
  },
  methods: {
    handleChange() {
      this.father.inputValue = this.inputValue;
    },
  },
};
</script>
<template>
  <div>
    <input type="text" v-model="inputValue" @change="handleChange" />
  </div>
</template>
<script>
export default {
  props: {
    inputValue: String,
  },
  created() {
    console.log(this.$attrs, "grandSon---$attrs");
    console.log(this.$listeners, "grandSon---$listeners");
  },
  methods: {
    handleChange() {
      this.father.inputValue = this.inputValue;
    },
  },
};
</script>

vue如何實現(xiàn)組件通信

通過這種方式,祖孫之間也實現(xiàn)了通信。

六、provide/inject

provide/inject可以在一個祖先組件中向它的所有后輩組件注入一個依賴,只要上下游關系成立就能生效。簡單的理解就是provide是注入數(shù)據(jù),inject是獲取數(shù)據(jù)。所以provide是用于父組件,inject是用于子孫組件。provide應該是一個對象或者返回一個對象的函數(shù),inject應該是一個字符串數(shù)組或者一個對象。官網(wǎng)提到這么一句話:

提示:provide 和 inject 綁定并不是可響應的。這是刻意為之的。然而,如果你傳入了一個可監(jiān)聽的對象,那么其對象的 property 還是可響應的。

這句話怎么理解呢?字面理解就是你要想在上下游傳遞的那個數(shù)據(jù)是可響應的,那么就應該以對象的形式傳遞,先試一下以基本數(shù)據(jù)類型的形式傳遞,看下例子:
父組件:

<template>
  <div>
    <p>父組件</p>
    <input type="text" v-model="inputValue" />
    <p>子組件</p>
    <Son></Son>
    <p>孫組件</p>
    <GrandSon></GrandSon>
  </div>
</template>
<script>
import Son from "./son.vue";
import GrandSon from "./grandSon.vue";
export default {
  components: { Son, GrandSon },
  provide() {
    return {
      father: this.inputValue,
    };
  },
  data() {
    return {
      inputValue: "123",
    };
  },
};
</script>

子組件:

<template>
  <div>
    <input type="text" v-model="inputValue" @change="handleChange" />
  </div>
</template>
<script>
export default {
  inject: ["father"],
  data() {
    return {
      inputValue: "",
    };
  },
  watch: {
    father(val) {
      console.log(val, "val");
      this.inputValue = val;
    },
  },
  created() {
    console.log(this, "this");
  },
  methods: {
    handleChange() {
      this.father.inputValue = this.inputValue;
    },
  },
};
</script>

在子組件打印this:

vue如何實現(xiàn)組件通信

可以看到,父組件的inputValue值是被注入到子組件當中的。但卻監(jiān)聽不到這個father。

vue如何實現(xiàn)組件通信

然后,我們改成以對象的形式進行注入:

<template>
  <div>
    <p>父組件</p>
    <input type="text" v-model="formData.inputValue" />
    <p>子組件</p>
    <Son></Son>
    <p>孫組件</p>
    <GrandSon></GrandSon>
  </div>
</template>
<script>
import Son from "./son.vue";
import GrandSon from "./grandSon.vue";
export default {
  components: { Son, GrandSon },
  provide() {
    return {
      father: this.formData,
    };
  },
  data() {
    return {
      formData: {
        inputValue: "123",
      },
    };
  },
};
</script>
<template>
  <div>
    <input type="text" v-model="inputValue" @change="handleChange" />
  </div>
</template>
<script>
export default {
  inject: ["father"],
  data() {
    return {
      inputValue: "",
    };
  },
  watch: {
    'father.inputValue'(val){
      console.log(val, "val");
      this.inputValue = val;
    },
  },
  created() {
    console.log(this, "this");
  },
  methods: {
    handleChange() {
      this.father.inputValue = this.inputValue;
    },
  },
};
</script>

這個時候我們看下打印的this以及效果:

vue如何實現(xiàn)組件通信

vue如何實現(xiàn)組件通信

這樣就可以實現(xiàn)數(shù)據(jù)的響應了。這里有一個點需要注意,如果在父組件中將整個父組件的this注入到后代組件中,在后代組件中是不能通過深度監(jiān)聽來監(jiān)聽這個注入的對象的,會報堆棧溢出的錯誤。所以這里我用的是this.formData的形式注入。這樣在子孫組件中可以通過'father.inputValue'這樣的形式監(jiān)聽,也可以通過這樣的形式:

father: {
      handler(val) {
        console.log(val);
      },
      deep: true,
    },

至于為什么會導致這個問題,我們先看下深度監(jiān)聽的實現(xiàn)方式:

vue如何實現(xiàn)組件通信

這段注釋什么意思呢,簡單理解就是vue是通過遞歸遍歷對象里面的每一個屬性,將是對象的屬性收集起來進行監(jiān)聽。眾所周知,遞歸是很容易引起堆棧溢出的,而看下this對象就不難理解為什么會導致堆棧溢出了(太多了,而且是層層嵌套下去的)。

以上是“vue如何實現(xiàn)組件通信”這篇文章的所有內(nèi)容,感謝各位的閱讀!希望分享的內(nèi)容對大家有幫助,更多相關知識,歡迎關注億速云行業(yè)資訊頻道!

向AI問一下細節(jié)

免責聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點不代表本網(wǎng)站立場,如果涉及侵權請聯(lián)系站長郵箱:is@yisu.com進行舉報,并提供相關證據(jù),一經(jīng)查實,將立刻刪除涉嫌侵權內(nèi)容。

vue
AI