Vous ne pouvez pas modifier directement la valeur de prop dans vue. Si vous modifiez directement vue, un avertissement sera généré, et la modification de la valeur de la propriété ne peut pas modifier la variable correspondant au composant parent cependant, vous pouvez déclencher l'événement du composant enfant, et le composant parent écoutera l'événement et l'exécutera. Modifiez la fonction du composant parent et réalisez une liaison de données bidirectionnelle en écoutant les événements du composant enfant pour modifier la valeur dans l'accessoire.
L'environnement d'exploitation de ce tutoriel : système Windows 10, version Vue3, ordinateur Dell G3.
Dans Vue, prop peut accepter les attributs transmis des composants parents aux composants enfants, mais prop ne peut pas être modifié.
Si un accessoire est modifié de force, il y aura un avertissement
Tous les accessoires forment une liaison descendante unidirectionnelle entre leurs accessoires parent et enfant : Les mises à jour des accessoires parent circuleront vers le bas vers les composants enfants, mais vice versa Non . Cela empêchera le composant enfant de modifier accidentellement l'état du composant parent, ce qui rendrait le flux de données de votre application difficile à comprendre.
De plus, chaque fois que le composant parent change, tous les accessoires du composant enfant seront actualisés avec les dernières valeurs. Cela signifie que vous ne devez pas modifier les accessoires à l'intérieur d'un composant enfant. Si vous faites cela, Vue émettra un avertissement dans la console du navigateur.
Essayez de modifier directement la valeur de l'attribut via la méthode suivante
this.$props.modelValue = true;
Vue émettra un avertissement
L'opération de définition sur la clé « modelValue » a échoué : la cible est en lecture seule.
Et la modification de la valeur de l'attribut ne peut pas modifier la variable correspondante au composant parent, s'applique uniquement aux types de données de base.
Notez qu'en JavaScript, les objets et les tableaux sont transmis par référence, donc pour un accessoire de type tableau ou objet, La modification de l'objet ou du tableau lui-même dans le composant enfant affectera l'état du composant parent, et Vue ne peut pas avertir vous à ce sujet . En règle générale, vous devez éviter de modifier des accessoires, y compris des objets et des tableaux, car cela ignore la liaison de données unidirectionnelle et peut conduire à des résultats inattendus.
Vue3 Props Document
La méthode de modification correcte consiste à déclencher l'événement du composant enfant. Le composant parent écoute l'événement et exécute la fonction qui modifie le composant parent. Prenez v-model
comme exemple. , v-model
lui-même est l'attribut modelValue
transmis au composant enfant, puis écoute l'événement update:modelValue
pour obtenir v-model
为例,v-model
本身就是传递给子组件的 modelValue
属性,然后监听 update:modelValue
事件来做到双向数据绑定,所以我们也可以通过该方法来在代码中修改 modelValue
this.$emit("update:modelValue", !this.$props.modelValue);
v-model
将自动监听 update:modelValue
事件并将 modelValue
修改为触发事件时传递的参数值(即 $emit 的第二个参数)。
需要注意的是,这种通过触发事件-监听事件的数据流模式并不能马上生效,如果使用以下代码
this.$emit("update:modelValue", !this.$props.modelValue);console.log(this.$props.modelValue);
会发现输出到控制台的仍然是 modelValue
原先的值,这是因为触发事件-监听事件的模式下属性值的修改需要时间,改用以下代码会发现输出正常:
this.$emit("update:modelValue", !this.$props.modelValue);setTimeout(()=> {console.log(this.$props.modelValue)}, 1000)
由于 Vant 只实现了左边放 label
的输入框,想基于此实现一个 label
在右边的输入框
Vant3 Field 文档
这其中遇到的问题除了把 label
右置并根据 label
的内容自适应宽度外,就是 Vant 中的 <van-field />
与我需要实现的 <right-label-input />
以及放置 <right-label-input>
的父组件之间的数据流了。
为了能够与 <van-field />
一致使用 v-model
,我在 <right-label-input />
下设置 modelValue
属性,然后将 <van-field>
的 v-model="modelValue"
<template> <van-field :placeholder="placeholder" :name="name" v-model="modelValue" :disabled="disabled"/> </template> <script> export default { name: "RightLabelInput", props: { modelValue: "", }, } </script>
无疑,这种做法违背了单向数据流原则,在 VanField
修改时将直接影响 RightLabelInput
的 modelValue
值,而 modelValue
是 RightLabelInput
deux- manière de liaison de données
modelValue
<template> <van-field :placeholder="placeholder" :name="name" v-model="input" :disabled="disabled" /> </template> <script> export default { name: "RightLabelInput", props: { placeholder: "", name: "", label: "", modelValue: "", disabled: { type: Boolean, default: false } }, data() { return { input: this.modelValue, // 绑定到 data 的变量,防止直接修改 Props } }, beforeMount() { this.input = this.modelValue }, watch: { input(newInput, oldInput) { this.$emit("update:modelValue", newInput); // 监听 input 在修改时发起事件,由父组件修改 modelValue 的值 }, modelValue(newValue, oldValue) { this.input = newValue; } } } </script> <style scoped> </style>
v-model
écoutera automatiquement le update:modelValue
événement et remplacez-le par modelValue
Modifier la valeur du paramètre transmise lorsque l'événement est déclenché (c'est-à-dire le deuxième paramètre de $emit). Il convient de noter que ce mode de flux de données de déclenchement d'événements et de surveillance des événements ne prend pas effet immédiatement si vous utilisez le code suivantrrreee
, vous constaterez que la sortie vers la console est toujoursmodelValuecode> valeur d'origine, cela est dû au fait qu'il faut du temps pour modifier la valeur de l'attribut en mode événement déclencheur-écoute. Si vous utilisez le code suivant, vous constaterez que la sortie est normale : <p>rrreee<a href="https://www.php.cn/vuejs/" target="_blank">Pratique</a></p>Depuis Vant uniquement. implémente <code>label
sur la zone de saisie de gauche, je souhaite implémenter un label
basé sur cette zone de saisie à droite🎜🎜Document Vant3 Field🎜🎜Les problèmes rencontrés dans ce domaine sont d'ailleurs en plaçant label
à droite et selon label, il s'agit du <van-field />
dans Vant et le <right-label-input /> que je dois implémenter. Les données circulent entre ;
et le composant parent où <right-label-input>
est. mis. 🎜🎜Mauvaise approche🎜🎜Pour pouvoir utiliser v-model
de manière cohérente avec <van-field />
, j'ai utilisé <right-label- input / >
Définissez l'attribut modelValue
sous <van-field>
🎜Cette approche viole sans aucun doute le principe du flux de données unidirectionnel lorsque <. code>VanField est modifié, cela affectera directement la valeur modelValue
de RightLabelInput
, et modelValue
est une propriété de RightLabelInput
et ne peut pas être modifié directement. 🎜🎜Peut-être la bonne approche🎜rrreee🎜🎜Mots clés🎜 : Flux de données unidirectionnel🎜🎜[Recommandation associée : "🎜tutoriel vue.js🎜"]🎜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!