Maison > Article > interface Web > Résumez et partagez les différentes manières de mettre en œuvre la communication entre les composants dans Vue, pour ne plus avoir peur des entretiens !
Intervieweur : De combien de façons disposez-vous pour mettre en œuvre la communication entre les composants Vue ? L'article suivant résumera et partagera avec vous plusieurs façons de mettre en œuvre la communication entre les composants Vue. Vous n'aurez plus peur des entretiens, j'espère qu'il vous sera utile !
La composantisation est l'une des idées importantes du framework Vue. Avant l'apparition du framework front-end, une page de site Web n'était généralement qu'un fichier. Si une page contenait des données que tout le monde devait utiliser, il suffisait de déclarer un fichier global. variable directement. Cependant, après l'émergence du framework Vue, une page a été composée de composants, ce qui signifie qu'une page a été divisée en plusieurs fichiers. Le partage de données entre les composants est alors devenu un gros problème. Bien sûr, Vue fournit beaucoup de choses pour réaliser le partage de données entre eux. Composants. Méthodes, nous allons aujourd'hui déterminer quelles sont les méthodes disponibles ? (Partage de vidéos d'apprentissage : tutoriel vuejs)
Parce qu'il n'y en a que quelques-uns couramment utilisés dans les projets, de nombreux amis ne parviennent souvent pas à tout expliquer lors des entretiens, il est donc recommandé de faire le tri.
Étant donné que tous les composants de Vue sont sous la forme d'une arborescence de composants, il existe de nombreuses situations de communication entre les composants, à peu près les suivantes :
Les méthodes de communication recommandées sont différentes dans chaque scénario. Vous devez choisir la méthode de communication la plus appropriée entre les composants en fonction des différents scénarios.
Cette méthode est généralement utilisée pour transmettre des valeurs entre les composants parent et enfant. Le composant parent transmet la valeur au composant enfant via des attributs, et le composant enfant la reçoit via des accessoires. Les composants enfants transmettent des données aux composants parents via des événements.
Projet d'initialisation :
Nous avons construit le projet Vue le plus simple et construit respectivement 3 composants : parent, child1, child2, puis avons introduit le composant parent dans APP.vue, et introduit l'enfant1 et l'enfant2 dans le composant parent. , l'interface d'exécution initiale est la suivante :
Ensuite, nous utilisons des attributs pour transmettre les valeurs du composant parent au composant enfant.
Exemple de code du composant parent :
// src/views/parent.vue <template> <div class="parent-box"> <p>父级组件</p> <div> <button @click="changeMsg">更改数据</button> </div> <child1 :msg="msg"></child1> <child2 :msg="msg"></child2> </div> </template> <script> import child1 from "./child1.vue"; import child2 from "./child2.vue"; export default { data() { return { msg: "我是父组件的数据", }; }, components: { child1, child2, }, methods: { // 点击按钮更改数据 changeMsg() { this.msg = "变成小猪课堂"; }, }, }; </script>
Nous transmettons le msg du composant parent au composant enfant via : msg="msg", et lorsque vous cliquez sur le bouton, le msg dans le composant parent sera modifié.
Exemple de code de sous-composant :
// src/views/child1.vue <template> <div class="child-1"> <p>child1组件</p> <div> <p>parent组件数据:{{ msg }}</p> </div> </div> </template> <script> export default { props: { msg: { type: String, default: "", }, }, }; </script>
Le sous-composant reçoit des données du composant parent via l'attribut props.
Résultat de sortie :
Lorsque nous cliquons sur le bouton, les données du composant parent changent et les données reçues par le composant enfant changent également.
Remarque : Le msg reçu par :msg="msg" est une variable Vous pouvez vous référer au principe d'utilisation de bind Si vous n'ajoutez pas :, le msg reçu est une chaîne.
Le composant enfant peut transmettre la valeur au composant parent via l'événement personnalisé $emit. Le composant parent doit écouter cet événement pour recevoir la valeur du composant enfant. .
Exemple de code du composant parent :
// src/views/parent.vue <template> <div class="parent-box"> <p>父级组件</p> <div> <button @click="changeMsg">更改数据</button> </div> <div>子组件数据:{{ childData }}</div> <child1 :msg="msg" @childData="childData"></child1> <child2 :msg="msg"></child2> </div> </template> <script> import child1 from "./child1.vue"; import child2 from "./child2.vue"; export default { data() { return { msg: "我是父组件的数据", childData: "", }; }, components: { child1, child2, }, methods: { changeMsg() { this.msg = "变成小猪课堂"; }, // 监听子组件事件 childData(data) { this.childData = data; }, }, }; </script>
Exemple de code du composant enfant :
// src/views/child1.vue <template> <div class="child-1"> <p>child1组件</p> <div> <button @click="sendData">传递数据给父组件</button> </div> <div> <p>parent组件数据:{{ msg }}</p> </div> </div> </template> <script> export default { props: { msg: { type: String, default: "", }, }, methods: { // 点击按钮,使用$emit向父组件传递数据 sendData() { this.$emit("childData", "我是子组件数据"); }, }, }; </script>
Résultat de sortie :
Nous écoutons l'événement childData via @childData="getChildData" dans le composant parent, pour obtenir les données transmises par le composant enfant, et l'événement $emit est déclenché en cliquant sur le bouton dans le composant enfant pour transmettre les données au composant parent. Lorsque nous cliquons sur le bouton "Transmettre les données au composant parent", le composant parent peut récupérer les données.
Cette méthode permet au composant enfant d'obtenir la valeur du composant parent de manière très pratique, y compris non seulement les données, mais également les méthodes.
Exemple de code de sous-composant :
// src/views/child1.vue <template> <div class="child-1"> <p>child1组件</p> <div> <button @click="sendData">传递数据给父组件</button> </div> <div> <button @click="getParentData">使用$parent</button> </div> <div> <p>parent组件数据:{{ msg }}</p> </div> </div> </template> <script> export default { props: { msg: { type: String, default: "", }, }, methods: { sendData() { this.$emit("childData", "我是子组件数据"); }, // 通过$parent方式获取父组件值 getParentData() { console.log("父组件", this.$parent); }, }, }; </script>
点击“使用parent获取父组件的属性或数据。
输出结果:
我们可以看到控制台打印出了父组件的所有属性,不仅仅包含了data数据,还有里面定义的一些方法等。
$children
和$refs
获取子组件值这两种方式和$parent非常的类似,它们可以直接获取子组件的相关属性或方法,不仅限于数据。
父组件示例代码:
// src/views/parent.vue <template> <div class="parent-box"> <p>父级组件</p> <div> <button @click="changeMsg">更改数据</button> </div> <div> <button @click="getChildByRef">使用$children和$refs</button> </div> <div>子组件数据:{{ childData }}</div> <child1 ref="child1" :msg="msg" @childData="getChildData"></child1> <child2 :msg="msg"></child2> </div> </template> <script> import child1 from "./child1.vue"; import child2 from "./child2.vue"; export default { data() { return { msg: "我是父组件的数据", childData: "", }; }, components: { child1, child2, }, methods: { changeMsg() { this.msg = "变成小猪课堂"; }, // 监听子组件的自定义事件 getChildData(data) { this.childData = data; }, // 使用$chilren和$refs获取子组件 getChildByRef() { console.log("使用$children", this.$children); console.log("使用$refs", this.$refs.child1); }, }, }; </script>
输出结果:
上段代码中,我们点击按钮,分别通过refs的方式获取到了子组件,从而拿到子组件数据。需要注意的是,refs时,需要在子组件上添加ref属性,有点类似于直接获取DOM节点的操作。
$attrs
和$listeners
$attrs
est nouvellement proposée après Vue2.4.0 et est généralement utilisée lorsque des composants multicouches transfèrent des données. Si de nombreux amis sont confrontés à un scénario de transfert de données de composants multicouches, ils peuvent directement choisir Vuex pour le transfert. Cependant, si les données que nous devons transférer n'impliquent pas de mises à jour et de modifications de données, il est recommandé d'utiliser la méthode $arrts. , Vuex est toujours relativement lourd.
Explication officielle du site Web :
inclut des liaisons d'attributs (sauf classe et style) qui ne sont pas reconnues (et obtenues) comme accessoires dans la portée parent. Lorsqu'un composant ne déclare aucun accessoire, toutes les liaisons de portée parent (sauf la classe et le style) seront incluses ici, et les composants internes peuvent être transmis via v-bind="$attrs" - lors de la création de composants de haut niveau. Très utile.
L'explication sur le site officiel est encore difficile à comprendre. Nous pouvons l'expliquer en termes plus populaires.
Explication populaire :
Lorsque le composant parent transmet beaucoup de données au composant enfant et que le composant enfant ne déclare pas d'accessoires pour les recevoir, alors le
说的再多可能还是没有代码来得简单易懂,我们新建一个孙子组件child1-child.vue,编写之后界面如下:
我们在parent父组件中多传一点数据给child1组件。
parent组件示例代码:
// src/views/parent.vue <template> <div class="parent-box"> <p>父级组件</p> <child1 ref="child1" :msg="msg" :msg1="msg1" :msg2="msg2" :msg3="msg3" :msg4="msg4" @childData="getChildData" ></child1> </div> </template> <script> import child1 from "./child1.vue"; import child2 from "./child2.vue"; export default { data() { return { msg: "我是父组件的数据", msg1: "parent数据1", msg2: "parent数据2", msg3: "parent数据3", msg4: "parent数据4", childData: "", }; }, components: { child1, child2, } }; </script>
这里我们删除了一些本节用不到的代码,大家需要注意一下。
child1组件示例代码:
// src/views/child1.vue <template> <div class="child-1"> <p>child1组件</p> <!-- 子组件child1-child --> <child1-child v-bind="$attrs"></child1-child> </div> </template> <script> import Child1Child from "./child1-child"; export default { components: { Child1Child, }, props: { msg: { type: String, default: "", }, }, mounted() { console.log("child1组件获取$attrs", this.$attrs); } }; </script>
输出结果:
上段代码中我们的parent父组件传递了5个数据给子组件:msg、msg1、msg2、msg3、msg4。但是在子组件中的props属性里面,我们只接收了msg。然后我们在子组件mounted中打印了$attrs,发现恰好少了props接收过的msg数据。
当我们在child1组件中使用attrs"的形式在传递给它的子组件child1-child,上段代码中我们已经加上了v-bind。
child1-child组件示例代码:
// src/views/child1-child.vue <template> <div class="child1-child"> <p>我是孙子组件child1-child</p> </div> </template> <script> export default { props: { msg1: { type: String, default: "", }, }, mounted() { console.log("child1-child组件$attrs", this.$attrs); }, }; </script>
输出结果:
我们发现child1-child组件中打印的$attrs中少了msg1,因为我们已经在props中接收了msg1。
attrs属性和类型,只是它们传递的东西不一样。
官网的解释:
包含了父作用域中的 (不含 .native 修饰器的) v-on 事件监听器。它可以通过 v-on="$listeners" 传入内部组件——在创建更高层次的组件时非常有用。
通俗的解释:
当父组件在子组件上定义了一些自定义的非原生事件时,在子组件内部可以通过$listeners属性获取到这些自定义事件。
它和attrs用来传递属性,$listeners用来传递非原生事件,我们在child1组件中打印一下看看。
child1组件示例代码:
// src/views/child1.vue mounted() { console.log("child1组件获取$attrs", this.$attrs); console.log("child1组件获取$listeners", this.$listeners); },
输出结果:
可以发现输出了childData方法,这是我们在它的父组件自定义的监听事件。除次之外,$listeners可以通过v-on的形式再次传递给下层组件。
child1组件示例代码:
// src/views/child1.vue <template> <div class="child-1"> <p>child1组件</p> <div> <button @click="sendData">传递数据给父组件</button> </div> <div> <button @click="getParentData">使用$parent</button> </div> <div> <p>parent组件数据:{{ msg }}</p> </div> <!-- 子组件child1-child --> <child1-child v-bind="$attrs" v-on="$listeners"></child1-child> </div> </template>
child1-child组件示例代码:
// src/views/child1-child.vue mounted() { console.log("child1-child组件$attrs", this.$attrs); console.log("child1-child组件$listerners", this.$listeners); },
输出结果:
可以看到在child1-child孙子组件中也获得了parent父组件中的childData自定义事件。使用emit的方式逐级向上触发事件,只需要使用$listerners就可以得到父组件中的自定义事件,相当于偷懒了。
可能细心的小伙伴会发现,我们在使用$attrs时,child1子组件渲染的DOM节点上将我们传递的属性一起渲染了出来,如下图所示:
这并不是我们想要的,为了解决这个问题,我们可以在子组件中设置inheritAttrs属性。
官网解释:
默认情况下父作用域的不被认作 props 的 attribute 绑定 (attribute bindings) 将会“回退”且作为普通的 HTML attribute 应用在子组件的根元素上。当撰写包裹一个目标元素或另一个组件的组件时,这可能不会总是符合预期行为。通过设置 inheritAttrs 到 false,这些默认行为将会被去掉。而通过 (同样是 2.4 新增的) 实例 property $attrs 可以让这些 attribute 生效,且可以通过 v-bind 显性的绑定到非根元素上。
官网说了非常多,但是不太通俗,我们可以简单的理解。
通俗解释:
父组件传递了很多数据给子组件,子组件的props没有完全接收,那么父组件传递的这些数据就会渲染到HTML上,我们可以给子组件设置inheritAttrs 为false,避免这样渲染。
child1组件示例代码:
// src/views/child1.vue props: { msg: { type: String, default: "", }, }, inheritAttrs: false,
输出结果:
此时我们节点上就没有那些无关的节点属性了。
在我们做项目的时候,会发现不相关的组件之间的数据传递是较为麻烦的,比如兄弟组件、跨级组件,在不使用Vuex情况下,我们可以使用自定义事件(也可以称作事件中心)的方式来实现数据传递。
事件中心的思想也比较简单:中间中心主要就两个作用:触发事件和监听事件。假如两个组件之间需要传递数据,组件A可以触发事件中心的事件,组件B监听事件中心的事件,从而让两个组件之间产生关联,实现数据传递。
实现步骤:
为了演示简单,我们在全局注册一个事件中心,修改main.js。
main.js代码如下:
// src/main.js Vue.config.productionTip = false Vue.prototype.$EventBus = new Vue() new Vue({ router, store, render: h => h(App) }).$mount('#app')
child1组件示例代码:
<template> <div class="child-1"> <p>child1组件</p> <div> <button @click="toChild2">向child2组件发送数据</button> </div> </div> </template> <script> import Child1Child from "./child1-child"; export default { methods: { // 通过事件总线向child2组件发送数据 toChild2() { this.$EventBus.$emit("sendMsg", "我是child1组件发来的数据"); }, }, }; </script>
child1组件中调用emit向事件中心添加sendMsg事件,这个用法有点类似与props和$emit的关系。
child2组件2示例代码:
// src/views/child1.vue <template> <div class="child-2"> <p>child2组件</p> <div> <p>parent组件数据:{{ msg }}</p> </div> </div> </template> <script> export default { props: { msg: { type: String, default: "", }, }, mounted() { this.$EventBus.$on("sendMsg", (msg) => { console.log("接收到child1发送来的数据", msg); }); }, }; </script>
当我们点击child1组件中的按钮时,就会触发sendMsg事件,在child2组件中我们监听了该事件,所以会接收到child1组件发来的数据。
输出结果:
事件中心实现数据传递的这种方式,其实就是一个发布者和订阅者的模式,这种方式可以实现任何组件之间的通信。
这两个是在Vue2.2.0新增的API,provide和inject需要在一起使用。它们也可以实现组件之间的数据通信,但是需要确保组件之间是父子关系。
官网的解释:
这对选项需要一起使用,以允许一个祖先组件向其所有子孙后代注入一个依赖,不论组件层次有多深,并在其上下游关系成立的时间里始终生效。
官网的解释就已经说得很明确了,所以这里我们就不需要通俗的解释了,简单一句话:父组件可以向子组件(无论层级)注入依赖,每个子组件都可以获得这个依赖,无论层级。
parent示例代码:
// src/views/parent.vue <script> import child1 from "./child1.vue"; import child2 from "./child2.vue"; export default { provide() { return { parentData: this.msg }; }, data() { return { msg: "我是父组件的数据", msg1: "parent数据1", msg2: "parent数据2", msg3: "parent数据3", msg4: "parent数据4", childData: "", }; }, components: { child1, child2, }, }; </script>
child1-child组件示例代码:
// src/views/child1-child.vue <template> <div class="child1-child"> <p>我是孙子组件child1-child</p> <p>parent组件数据:{{parentData}}</p> </div> </template> <script> export default { inject: ["parentData"], props: { msg1: { type: String, default: "", }, }, mounted() { console.log("child1-child组件$attrs", this.$attrs); console.log("child1-child组件$listerners", this.$listeners); console.log("child1-child组件获取parent组件数据", this.parentData) }, }; </script>
输出结果:
通过provide和inject结合的方式,我们在child1-child组件中获取到了parent组件中的数据。如果你下来尝试过的话,可能会发现一个问题,此时数据不是响应式,也就是parent组件更改了数据,child1-child组件中的数据不会更新。
想要变为响应式的,我们需要修改一下provide传递的方式。
parent代码如下:
// src/views/parent.vue <script> import child1 from "./child1.vue"; import child2 from "./child2.vue"; export default { provide() { return { parentData: this.getMsg }; }, data() { return { msg: "我是父组件的数据", msg1: "parent数据1", msg2: "parent数据2", msg3: "parent数据3", msg4: "parent数据4", childData: "", }; }, components: { child1, child2, }, methods: { // 返回data数据 getMsg() { return this.msg; }, }, }; </script>
这个时候我们会发现数据变为响应式的了。
porvide和inject的原理可以参考下图:
这两种方式应该是小伙伴们在实际项目中使用最多的了,所以这里就不但展开细说,只是提一下这两者的区别即可。
Vuex:
localstorage:
v-model是vue中的一个内置指令,它通常用在表单元素上以此来实现数据的双向绑定,它的本质是v-on和v-bind的语法糖。在这里我们也可以借助它来实现某些场景下的数据传递。注意,这儿的场景必须是父子组件。
parent组件示例代码:
<template> <div class="parent-box"> <p>父级组件</p> <div>modelData: {{modelData}}</div> <child2 :msg="msg" v-model="modelData"></child2> <!-- 实际等同于 --> <!-- <child2 v-bind:value="modelData" v-on:input="modelData=$event"></child2> --> </div> </template> <script> import child2 from "./child2.vue"; export default { provide() { return { parentData: this.getMsg }; }, data() { return { modelData: "parent组件的model数据" }; }, components: { child1, }, }; </script>
child2组件示例代码:
<template> <div class="child-2"> <p>child2组件</p> <div> <button @click="confirm">修改v-model数据</button> </div> </div> </template> <script> export default { props: { value: { type: String, default: "", }, }, mounted() { console.log("child2组件接收附件见v-model传递的数据", this.value); }, methods: { // 通过$emit触发父组件的input事件,并将第二个参数作为值传递给父组件 confirm() { this.$emit("input", "修改parent传递的v-model数据"); }, }, }; </script>
我们在父组件中使用v-model向child2子组件传递数据,子组件的props中使用默认的value属性接收,在子组件中利用$emit触发父组件中默认input事件,此时传递的数据便会在子组件和父组件中发生变化,这就是数据双向绑定。
如果想要更加详细的学习v-model的使用,可以参考官网。
Vue中组件通讯的方式有很多种,每一种应用的场景可能都有一些不一样,我们需要在合适的场景下选择合适的通讯方式。
, Introduction à la programmation)
Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!