vue组件之间相互传值的方式

本文详细介绍了Vue中组件间的通信方式,包括父组件向子组件传值(通过props和watch)、子组件向父组件传值(通过$emit和callback函数)、组件间通信的其他方法(如EventBus、Vuex、provide/inject)以及多层组件通信。强调了单向数据流原则和数据响应性,并提供了各种场景下的示例代码。

摘要生成于 C知道 ,由 DeepSeek-R1 满血版支持, 前往体验 >

Vue 作为一个轻量级的前端框架,其核心就是组件化开发。组件化是它的精髓,也是最强大的功能之一。而组件实例的作用域是相互独立的,这就意味着不同组件之间的数据无法相互引用。在开发过程中,我们需要访问其他组件的数据,组件之间的关系有:父子、兄弟、隔代

一、父组件向子组件传值

即父组件通过属性的方式向子组件传值,子组件通过 props 来接收。在父组件的子组件标签中绑定自定义属性

// 父组件
<user-detail :myName="name" />
    
export default {
    components: {
        UserDetail
    }
    ......
}

在子组件中使用props(可以是数组也可以是对象)接收即可。可以传多个属性。

// 子组件
export default {
    props: ['myName']
}/*
props: { myName: String } //这样指定传入的类型,如果类型不对会警告
props: { myName: [String, Number] } // 多个可能的类型
prosp: { myName: { type: String, requires: true } } //必填的的字符串
props: { 
    childMsg: { 
        type: Array, 
        default: () => [] 
    }
}  // default指定默认值
如果 props 验证失败,会在控制台发出一个警告。
*/

子组件接收的父组件的值分为引用类型普通类型两种:
普通类型:字符串(String)、数字(Number)、布尔值(Boolean)、空(Null)
引用类型:数组(Array)、对象(Object)

基于 vue 的单向数据流,即组件之间的数据是单向流通的,子组件是不允许直接对父组件传来的值进行修改的,所以应该避免这种直接修改父组件传过来的值的操作,否则控制台会报错。

如果传过来的值是简单数据类型,是可以在子组件中修改,也不会影响其他兄弟组件内同样调用了来自该父组件的值。

具体操作是可以先把传过来的值重新赋值给data中的一个变量,然后再更改那个变量

// 子组件
export default {
    props: ['myName'],
    data() {
        return {
            name : this.myName    // 把传过来的值赋值给新的变量
        }
    },
    watch: {
        myName(newVal) {
            this.name = newVal //对父组件传过来的值进行监听,如果改变也对子组件内部的值进行改变
        }
    },
    methods: {
        changeName() {  
            this.name = 'Lily'  // 这里修改的只是自己内部的值,就不会报错了
        },
    }
}

注:如果不使用 watch 来监听父组件传递的 myName 值,子组件中的 name 值是不会随着父组件的 myName 值进行改变,因为 data 中 name: this.myName 仅仅只是定义了一个初始值。
如果引用类型的值,当在子组件中修改后,父组件的也会修改,因其数据是公用的,其他同样引用了该值的子组件也会跟着被修改。可以理解成父组件传递给子组件的值,就相当于复制了一个副本,这个副本的指针还是指向父组件中的那个,即共享同一个引用。所以除非有特殊需要,否则不要轻易修改。

二、子组件向父组件传值

2.1 子组件绑定一个事件,通过 this.$emit() 来触发

在子组件中绑定一个事件,并给这个事件定义一个函数

// 子组件
<button @click="changeParentName">改变父组件的name</button>
​
export default {
    methods: {
        //子组件的事件
        changeParentName: function() {
            this.$emit('handleChange', 'Jack') // 触发父组件中handleChange事件并传参Jack
            // 注:此处事件名称与父组件中绑定的事件名称要一致
        }
    }
}

在父组件中定义并绑定 handleChange 事件

// 父组件
<child @handleChange="changeName"></child>
​
methods: {
    changeName(name) {  // name形参是子组件中传入的值Jack
        this.name = name
    }
}

2.2. 通过 callback 函数

先在父组件中定义一个callback函数,并把 callback 函数传过去

// 父组件
<child :callback="callback"></child>
​
methods: {
    callback: function(name) {
        this.name = name
    }
}

在子组件中接收,并执行 callback 函数

// 子组件
<button @click="callback('Jack')">改变父组件的name</button>
​
props: {
    callback: Function,
}

2.3 通过 $parent / $children 或 $refs 访问组件实例

这两种都是直接得到组件实例,使用后可以直接调用组件的方法或访问数据。

// 子组件
export default {
  data () {
    return {
      title: '子组件'
    }
  },
  methods: {
    sayHello () {
        console.log('Hello');
    }
  }
}
// 父组件
<template>
  <child ref="childRef" />
</template><script>
  export default {
    created () {
      // 通过 $ref 来访问子组件
      console.log(this.$refs.childRef.title);  // 子组件
      this.$refs.childRef.sayHello(); // Hello
      
      // 通过 $children 来调用子组件的方法
      this.$children.sayHello(); // Hello 
    }
  }
</script>

注:这种方式的组件通信不能跨级。

2.4. $attrs / $listeners 详解点这里

三、兄弟组件之间传值

3.1 还是通过 $emit 和 props 结合的方式

在父组件中给要传值的两个兄弟组件都绑定要传的变量,并定义事件

// 父组件
<child-a :myName="name" />
<child-b :myName="name" @changeName="editName" />  
    
export default {
    data() {
        return {
            name: 'John'
        }
    },
    components: {
        'child-a': ChildA,
        'child-b': ChildB,
    },
    methods: {
        editName(name) {
            this.name = name
        },
    }
}

在子组件B中接收变量和绑定触发事件

// child-b 组件
<p>姓名:{{ myName }}</p>
<button @click="changeName">修改姓名</button>
    
<script>
export default {
    props: ["myName"],
    methods: {
        changeName() {
            this.$emit('changeName', 'Lily')   // 触发事件并传值
        }
    }
}
</script>

子组件A

// child-a 组件
<p>姓名:{{ newName }}</p>
    
<script>
export default {
    props: ["myName"],
    computed: {
        newName() {
            if(this.myName) { // 判断是否有值传过来
                return this.myName
            }
            return 'John' //没有传值的默认值
        }
    }
}
</script>

即:当子组件B 通过 $emit() 触发了父组件的事件函数 editName,改变了父组件的变量name 值,父组件又可以把改变了的值通过 props 传递给子组件A,从而实现兄弟组件间数据传递。

3.2 通过一个空 vue 实例

创建一个 EventBus.js 文件,并暴露一个 vue 实例

import Vue from 'Vue'
export default new Vue()
//在要传值的文件里导入这个空 vue 实例,绑定事件并通过 $emit 触发事件函数
//(也可以在 main.js 中全局引入该 js 文件,我一般在需要使用到的组件中引入)

<template>
    <div>
        <p>姓名: {{ name }}</p>
        <button @click="changeName">修改姓名</button>
    </div>
</template><script>
import { EventBus } from "../EventBus.js"
​
export default {
 data() {
     return {
         name: 'John',
     }
  },
  methods: {
      changeName() {
          this.name = 'Lily'
          EventBus.$emit("editName", this.name) // 触发全局事件,并且把改变后的值传入事件函数
      }
    }
}
</script>

在接收传值的组件中也导入 vue 实例,通过 $on 监听回调,回调函数接收所有触发事件时传入的参数

import { EventBus } from "../EventBus.js"
​
export default {
    data() {
        return {
            name: ''
        }
    },
    created() {
         EventBus.$on('editName', (name) => {
             this.name = name
         })
    }
}

这种通过创建一个空的 vue 实例的方式,相当于创建了一个事件中心或者说是中转站,用来传递和接收事件。这种方式同样适用于任何组件间的通信,包括父子、兄弟、跨级,对于通信需求简单的项目比较方便,但对于更复杂的情况,或者项目比较大时,可以使用 vue 提供的更复杂的状态管理模式 Vuex 来进行处理。

3.3 使用 vuex →点这里

四、多层父子组件通信

有时需要实现通信的两个组件不是直接的父子组件,而是祖父和孙子,或者是跨越了更多层级的父子组件,这种时候就不可能由子组件一级一级的向上传递参数,特别是在组件层级比较深,嵌套比较多的情况下,需要传递的事件和属性较多,会导致代码很混乱。

这时就需要用到 vue 提供的更高阶的方法:provide/inject。

provide/inject 这对选项需要一起使用,以允许一个祖先组件向其所有子孙后代注入一个依赖,不论组件层次有多深,并在起上下游关系成立的时间里始终生效。查看官网

provide/inject:简单来说就是在父组件中通过provider来提供变量,然后在子组件中通过inject来注入变量,不管组件层级有多深,在父组件生效的生命周期内,这个变量就一直有效。

//父组件:
export default {
  provide: { // 它的作用就是将 **name** 这个变量提供给它的所有子组件。
    name: 'Jack'
  }
}
//子组件:
export default {
  inject: ['name'], // 注入了从父组件中提供的name变量
  mounted () {
    console.log(this.name);  // Jack
  }
}

provide 和 inject 绑定并不是可响应的。即父组件的name变化后,子组件不会跟着变。

如果想要实现 provide 和 inject 数据响应,有两种方法:

provide 祖先组件的实例,然后在子孙组件中注入依赖,这样就可以在后代组件中直接修改祖先组件的实例的属性,不过这种方法有个缺点就是这个实例上挂载很多没有必要的东西比如 props,methods

// 父组件 
<div>
      <button @click="changeName">修改姓名</button>
      <child-b />
</div>
<script>
    ......
    data() {
        return {
            name: "Jack"
        };
    },
    provide() {
        return {
            parentObj: this //提供祖先组件的实例
        };
    },
    methods: {
        changeName() {
            this.name = 'Lily'
        }
    }
</script>
    
//后代组件中取值:  
<template>
  <div class="border2">
    <P>姓名:{{parentObj.name}}</P>
  </div>
</template>
<script>
  export default {
    inject: {
      parentObj: {
        default: () => ({})
      }
    } // 或者inject: ['parentObj']
  };
</script>

注:这种方式在函数式组件中用的比较多。函数式组件,即无状态(没有响应式数据),无实例化(没有 this上下文),内部也没有任何生命周期处理方法,所以渲染性能高,比较适合依赖外部数据传递而变化的组件。

使用 Vue.observable 优化响应式 provide,这个我用的不熟就不说了,可以 → 官方文档

总结

父子通信:
父向子 传递数据是通过 props
子向父 是通过 $emit;通过 $parent / $children 通信; $ref也可以访问组件实例; provide / inject$attrs / $listeners

兄弟通信: EventBusVuex

跨级通信: EventBusVuexprovide / inject$attrs / $listeners

### Vue3 中自定义组件的方法 在 Vue3 中,父子组件间的主要依赖于 `props` 和 `$emit` 方法;而兄弟组件或跨级组件之间的通信则通常通过事件总线、Provide/Inject 或 Vuex 等状态管理工具来实现。以下是详细的介绍和示例。 #### 1. **父组件向子组件** 父组件可以通过 `props` 将数据传递给子组件。子组件接收这些数据并通过模板渲染出来。 ```html <!-- 父组件 --> <template> <div> <ChildComponent :message="parentMessage" /> </div> </template> <script> import ChildComponent from './ChildComponent.vue'; export default { components: { ChildComponent }, data() { return { parentMessage: 'Hello from parent component' }; } }; </script> ``` 子组件需要声明接受的 `props`: ```html <!-- 子组件 --> <template> <div>接收到的消息: {{ message }}</div> </template> <script setup> defineProps(['message']); </script> ``` 这种方法适用于简单的父子组件通信[^3]。 --- #### 2. **子组件向父组件** 子组件可以通过 `$emit` 向父组件发送事件,并携带参数。父组件监听该事件即可获取子组件的数据。 ```html <!-- 子组件 --> <template> <button @click="sendDataToParent">点击我向父组件发消息</button> </template> <script setup> const emit = defineEmits(['childEvent']); function sendDataToParent() { emit('childEvent', '这是来自子组件的消息'); } </script> ``` 父组件监听子组件发出的事件: ```html <!-- 父组件 --> <template> <div> <ChildComponent @childEvent="handleChildEvent" /> <p>从子组件接收到的消息: {{ childMessage }}</p> </div> </template> <script> import ChildComponent from './ChildComponent.vue'; export default { components: { ChildComponent }, data() { return { childMessage: '' }; }, methods: { handleChildEvent(message) { this.childMessage = message; } } }; </script> ``` 这种方式适合处理单向数据流中的回调需求[^2]。 --- #### 3. **双向绑定(v-model 增强)** Vue3 支持更灵活的 `v-model` 组件绑定方式,允许指定多个模型字段。 ```html <!-- 父组件 --> <template> <Child v-model:username="user.name" v-model:age="user.age" /> </template> <script> import Child from './Child.vue'; export default { components: { Child }, data() { return { user: { name: 'John Doe', age: 25 } }; } }; </script> ``` 子组件需定义对应的 `props` 和 `emits`: ```html <!-- 子组件 --> <template> <div> 用户名:<input type="text" :value="username" @input="updateUsername($event)" /><br/> 年龄:<input type="number" :value="age" @input="updateAge($event)" /> </div> </template> <script setup> const props = defineProps(['username', 'age']); const emit = defineEmits(['update:username', 'update:age']); function updateUsername(event) { emit('update:username', event.target.value); } function updateAge(event) { emit('update:age', Number(event.target.value)); } </script> ``` 这种模式特别适合表单类交互场景[^4]。 --- #### 4. **跨级组件通信(Provide/Inject)** 当涉及祖孙关系或多层嵌套时,可以使用 `provide/inject` 提供一种轻量级的状态共享方案。 ##### Provide 方式: ```javascript // 祖先组件 <script setup> import { provide, ref } from 'vue'; const sharedState = ref('祖先组件提供的数据'); provide('sharedKey', sharedState); </script> ``` ##### Inject 方式: ```javascript // 孙子组件 <script setup> import { inject } from 'vue'; const sharedState = inject('sharedKey'); // 获取祖先组件提供的数据 </script> ``` 此方法不破坏组件封装性的同时实现了深层次通信[^4]。 --- #### 5. **兄弟组件通信(事件总线或其他工具)** 如果两个无直接关联的组件需要相互通信,推荐使用第三方库如 Pinia/Vuex 或者手动创建一个事件中心。 简单例子如下: ```javascript // 创建事件总线 const bus = new Vue(); // 发送事件 bus.$emit('eventName', payload); // 接收事件 bus.$on('eventName', callbackFunction); ``` 现代开发建议优先采用集中化状态管理模式替代统事件总线[^4]。 --- ### 总结 以上介绍了几种常见的 Vue3 自定义组件方式及其适用范围,开发者应根据具体业务需求选取最合适的策略。
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值