Vue组件之间的通信

一、通信方式

  • Props 和 Events :通过父组件传递 props 给子组件,子组件使用**$emit** 发送事件到父组件。
  • Event Bus:使用一个中央事件总线来跨组件通信。
  • Vuex:使用 Vuex 进行全局状态管理,以便在任何组件间共享状态。
  • Provide / Inject :祖先组件使用 provide 传递数据,后代组件通过 **inject**接收数据。
  • $refs :父组件通过 **ref**获取子组件实例,直接调用其方法或访问属性。
  • Scoped Slots:使用插槽在父组件中传递数据到子组件,允许子组件根据传递的数据进行渲染。

二、应用举例

1、Props 和 Events

父组件 (Parent.vue)

javascript 复制代码
<template>
  <Child :message="parentMessage" @update="handleUpdate" />
</template>
<script>
export default {
  data() {
    return { parentMessage: 'Hello from parent' };
  },
  methods: {
    handleUpdate(newMessage) {
      this.parentMessage = newMessage;
    }
  }
}
</script>

子组件 (Child.vue)

javascript 复制代码
<template>
  <button @click="updateParent">Update Parent</button>
</template>
<script>
export default {
  props: ['message'],
  methods: {
    updateParent() {
      this.$emit('update', 'Hello from child');
    }
  }
}
</script>

2、Event Bus

事件总线 (eventBus.js)

javascript 复制代码
import Vue from 'vue';
export const EventBus = new Vue();

发送消息 (ComponentA.vue)

javascript 复制代码
<template>
  <button @click="sendMessage">Send Message</button>
</template>
<script>
import { EventBus } from './eventBus';
export default {
  methods: {
    sendMessage() {
      EventBus.$emit('message', 'Hello from Component A');
    }
  }
}
</script>

接收消息 (ComponentB.vue)

javascript 复制代码
<template>
  <p>{{ receivedMessage }}</p>
</template>
<script>
import { EventBus } from './eventBus';
export default {
  data() {
    return { receivedMessage: '' };
  },
  created() {
    EventBus.$on('message', (message) => {
      this.receivedMessage = message;
    });
  }
}
</script>

3、Vuex

Vuex Store (store.js)

javascript 复制代码
import Vue from 'vue';
import Vuex from 'vuex';

Vue.use(Vuex);

export default new Vuex.Store({
  state: {
    message: 'Hello from Vuex'
  },
  mutations: {
    setMessage(state, newMessage) {
      state.message = newMessage;
    }
  }
});

更新消息 (ComponentA.vue)

javascript 复制代码
<template>
  <button @click="updateMessage">Update Message</button>
</template>
<script>
import { mapMutations } from 'vuex';
export default {
  methods: {
    ...mapMutations(['setMessage']),
    updateMessage() {
      this.setMessage('Hello from Component A');
    }
  }
}
</script>

显示消息 (ComponentB.vue)

javascript 复制代码
<template>
  <p>{{ message }}</p>
</template>
<script>
import { mapState } from 'vuex';
export default {
  computed: {
    ...mapState(['message'])
  }
}
</script>

4、Provide / Inject

提供数据 (Ancestor.vue)

javascript 复制代码
<template>
  <Descendant />
</template>
<script>
export default {
  provide() {
    return {
      message: 'Hello from ancestor'
    };
  }
}
</script>

注入数据 (Descendant.vue)

javascript 复制代码
<template>
  <p>{{ message }}</p>
</template>
<script>
export default {
  inject: ['message']
}
</script>

5、$refs

父组件 (Parent.vue)

javascript 复制代码
<template>
  <Child ref="childComponent" />
  <button @click="callChildMethod">Call Child Method</button>
</template>
<script>
import Child from './Child.vue';
export default {
  components: { Child },
  methods: {
    callChildMethod() {
      this.$refs.childComponent.someMethod();
    }
  }
}
</script>

子组件 (Child.vue)

javascript 复制代码
<template>
  <p>Child Component</p>
</template>
<script>
export default {
  methods: {
    someMethod() {
      console.log('Method in child called!');
    }
  }
}
</script>

6、Scoped Slots

父组件 (Parent.vue)

javascript 复制代码
<template>
  <Child>
    <template v-slot:default="slotProps">
      <p>{{ slotProps.message }}</p>
    </template>
  </Child>
</template>
<script>
import Child from './Child.vue';
export default {
  components: { Child }
}
</script>

子组件 (Child.vue)

javascript 复制代码
<template>
  <slot :message="'Hello from child'"></slot>
</template>
<script>
export default {}
</script>
相关推荐
一点一木20 小时前
深度体验TRAE SOLO移动端7天:作为独立开发者,我把工作流揣进了兜里
前端·人工智能·trae
天外飞雨道沧桑21 小时前
TypeScript 中 omit 和 record 用法
前端·javascript·typescript
Lee川21 小时前
mini-cursor 揭秘:从 Tool 定义到 Agent 循环的完整实现
前端·人工智能·后端
canonical_entropy1 天前
从 Spec-Driven Development 到 Attractor-Guided Engineering
前端·aigc·ai编程
研☆香1 天前
聊聊前端页面的三种长度单位
前端
给钱,谢谢!1 天前
React + PixiJS 实现果园成长页:从状态机到浇水动画
前端·react.js·前端框架
暗冰ཏོ1 天前
VUE面试题大全
前端·javascript·vue.js·面试
次元工程师!1 天前
LangFlow开发(三)—Bundles组件架构设计(3W+字详细讲解)
java·前端·python·低代码·langflow
Bug-制造者1 天前
现代Web应用全栈开发:从架构设计到部署落地实战
前端
青春喂了后端1 天前
IntelliGit 前端状态层重构:把一个全局 Store 拆成清晰的状态边界
前端·重构·状态模式