vue组件传参的八种方式详细总结

news2024/11/13 10:37:50

在Vue中,组件之间的传参是构建动态和交互性用户界面的关键。Vue提供了多种方式来传递参数,以下是对这些方式的详细说明:

一、Props

Props是Vue中组件之间传递数据的一种常见方式。父组件可以通过props将数据传递给子组件,子组件通过props选项来接收这些数据。

  • 使用方式

    1. 父组件中定义要传递的数据,并在使用子组件时以属性的形式传递。
    2. 子组件中通过props选项来声明要接收的数据。
  • 示例

<!-- 父组件 -->
<template>
  <div>
    <child-component :message="parentMessage"></child-component>
  </div>
</template>

<script>
import ChildComponent from './ChildComponent.vue';

export default {
  components: {
    ChildComponent
  },
  data() {
    return {
      parentMessage: '这是来自父组件的消息'
    };
  }
};
</script>

<!-- 子组件 -->
<template>
  <div>{{ message }}</div>
</template>

<script>
export default {
  props: ['message']
};
</script>

二、$emit

emit是Vue中组件之间通过事件进行数据传递的一种方式。子组件可以通过emit方法触发一个自定义事件,并将数据传递给父组件。父组件可以通过在子组件标签上监听这个自定义事件来接收数据。

  • 使用方式

    1. 子组件中定义要触发的事件,并通过$emit方法传递数据。
    2. 父组件中在子组件标签上监听该事件,并定义处理函数来接收数据。
  • 示例

<!-- 子组件 -->
<template>
  <button @click="sendMessage">发送消息给父组件</button>
</template>

<script>
export default {
  methods: {
    sendMessage() {
      this.$emit('childMessage', '这是来自子组件的消息');
    }
  }
};
</script>

<!-- 父组件 -->
<template>
  <div>
    <child-component @childMessage="handleChildMessage"></child-component>
  </div>
</template>

<script>
import ChildComponent from './ChildComponent.vue';

export default {
  components: {
    ChildComponent
  },
  methods: {
    handleChildMessage(message) {
      console.log(message);
    }
  }
};
</script>


三、Provide/Inject

Provide/Inject是Vue中组件之间通过依赖注入进行数据传递的一种方式。父组件可以通过provide选项提供数据,子组件(包括跨层级的子孙组件)可以通过inject选项注入这些数据。

  • 使用方式

    1. 父组件中通过provide选项提供数据。
    2. 子组件中通过inject选项注入所需的数据。
  • 示例

<!-- 父组件 -->
<template>
  <div>
    <child-component></child-component>
  </div>
</template>

<script>
import ChildComponent from './ChildComponent.vue';

export default {
  components: {
    ChildComponent
  },
  provide() {
    return {
      message: '这是通过provide/inject传递的消息'
    };
  }
};
</script>

<!-- 子组件 -->
<template>
  <div>{{ message }}</div>
</template>

<script>
export default {
  inject: ['message']
};
</script>

四、attrs和listeners

$attrs
  • 概述:$attrs是一个对象,它包含了父作用域中没有被prop接收的所有属性(不包含class和style属性)。可以通过v-bind="$attrs"直接将这些属性传入内部组件,实现父组件隔代向孙组件传值。

  • 举例:父组件将nameage属性传递给子组件,子组件通过v-bind="$attrs"将这些属性(以及可能的其他未声明的属性)传递给孙组件。孙组件通过props接收这些属性。

<!-- 父组件(Parent.vue) -->
<template>
  <div>
    <h1>父组件</h1>
    <ChildComponent :name="parentName" :age="parentAge" />
  </div>
</template>

<script>
import ChildComponent from './ChildComponent.vue';
export default {
  components: { ChildComponent },
  data() {
    return {
      parentName: 'Tom',
      parentAge: 30
    };
  }
};
</script>

<!-- 子组件(ChildComponent.vue) -->
<template>
  <div>
    <h2>子组件</h2>
    <GrandChildComponent v-bind="$attrs" />
  </div>
</template>

<script>
import GrandChildComponent from './GrandChildComponent.vue';
export default {
  components: { GrandChildComponent }
};
</script>

<!-- 孙组件(GrandChildComponent.vue) -->
<template>
  <div>
    <h3>孙组件</h3>
    <p>父组件传递的名字:{{ name }}</p>
    <p>父组件传递的年龄:{{ age }}</p>
  </div>
</template>

<script>
export default {
  props: ['name', 'age']
};
</script>
$listeners
  • 概述:$listeners是一个对象,它包含了父组件中所有的v-on事件监听器(不包含.native修饰器的)。可以通过v-on="$listeners"将这些事件监听器传入内部组件,实现孙组件隔代向父组件传值。

  • 举例:

    在上述例子中,如果孙组件需要向父组件发送事件,可以通过$emit触发事件,并在子组件中使用v-on="$listeners"将这些事件传递给父组件。然而,需要注意的是,$listeners通常用于孙组件向隔代的父组件发送事件,而不是直接用于父子组件间的通信。在实际应用中,父子组件间的通信更多地使用$emitv-on

五、parent和children

parent和children是Vue中组件之间通过访问父组件和子组件实例进行数据传递的一种方式。子组件可以通过parent属性访问父组件实例,父组件可以通过children属性访问子组件实例(注意:$children是一个数组,包含了所有子组件的实例,但不保证顺序)。

  • 使用方式

    1. 通过parent或children访问相应的组件实例。
    2. 直接在组件实例上访问或修改数据。
  • 举例

注意:通常不建议直接使用$children进行组件间的通信,因为它可能导致代码难以维护和理解。如果需要访问子组件的数据或方法,更推荐使用refprovide/inject等机制。

<!-- 父组件(Parent.vue) -->
<template>
  <div>
    <h1>父组件</h1>
    <ChildComponent />
  </div>
</template>

<script>
import ChildComponent from './ChildComponent.vue';
export default {
  components: { ChildComponent },
  data() {
    return {
      parentMessage: 'Hello from Parent'
    };
  },
  methods: {
    parentMethod() {
      console.log('Parent method called');
    }
  }
};
</script>

<!-- 子组件(ChildComponent.vue) -->
<template>
  <div>
    <h2>子组件</h2>
    <button @click="callParentMethod">调用父组件方法</button>
  </div>
</template>

<script>
export default {
  methods: {
    callParentMethod() {
      this.$parent.parentMethod();
    }
  }
};
</script>

六、Vuex

Vuex是Vue中一种专门用于状态管理的插件。通过在Vuex中定义全局的状态,并在组件中使用getter和mutation来访问和修改状态,可以实现组件之间的数据传递和共享。

  • 使用方式

    1. 安装Vuex并创建store。
    2. 在store中定义状态、mutation、action等。
    3. 在组件中通过this.$store访问store,并使用getter获取状态,使用mutation或action修改状态。
  • 举例
<!-- 父组件(Parent.vue) -->
<template>
  <div>
    <h1>父组件</h1>
    <ChildComponent />
  </div>
</template>

<script>
import { mapState, mapMutations } from 'vuex';
import ChildComponent from './ChildComponent.vue';
export default {
  components: { ChildComponent },
  computed: {
    ...mapState(['counter'])
  },
  methods: {
    ...mapMutations(['increment'])
  }
};
</script>

<!-- 子组件(ChildComponent.vue) -->
<template>
  <div>
    <h2>子组件</h2>
    <p>计数器:{{ counter }}</p>
    <button @click="increment">增加</button>
  </div>
</template>

<script>
import { mapState, mapMutations } from 'vuex';
export default {
  computed: {
    ...mapState(['counter'])
  },
  methods: {
    ...mapMutations(['increment'])
  }
};
</script>

<!-- Vuex Store(store.js) -->
import Vue from 'vue';
import Vuex from 'vuex';

Vue.use(Vuex);

export default new Vuex.Store({
  state: {
    counter: 0
  },
  mutations: {
    increment(state) {
      state.counter++;
    }
  }
});

七、插槽(Slot)

        插槽是一种让父组件能够向子组件指定内容插入点的机制。通过插槽,父组件可以将自己的模板内容传递给子组件,并在子组件的指定位置渲染出来。

        插槽分为默认插槽、具名插槽和作用域插槽三种类型。

  • 使用方式

    1. 在子组件中定义插槽。
    2. 在父组件中使用子组件时,通过插槽向子组件传递内容。
  • 举例
默认插槽

默认插槽是最基本的插槽类型,用于在组件内传递和显示任意内容。如果没有给插槽命名,Vue会将内容传递到默认插槽中。

//子组件
<template>
  <div class="my-component">
    <slot></slot> <!-- 默认插槽 -->
  </div>
</template>
//父组件
<template>
  <MyComponent>
    <p>This is some default slot content!</p>
  </MyComponent>
</template>
具名插槽

具名插槽允许我们在组件中定义多个插槽,每个插槽都有一个唯一的名称。这样可以在组件中更精确地控制内容的显示位置。

//子组件(MyComponent.vue)
<template>
  <div class="my-component">
    <header>
      <slot name="header"></slot>
    </header>
    <main>
      <slot></slot> <!-- 默认插槽 -->
    </main>
    <footer>
      <slot name="footer"></slot><!--具名插槽-->
    </footer>
  </div>
</template>
//父组件
<template>
  <MyComponent>
    <template v-slot:header>
      <h1>Header Content</h1>
    </template>
    <p>This is some default slot content!</p>
    <template v-slot:footer>
      <p>Footer Content</p>
    </template>
  </MyComponent>
</template>

在这个例子中,<h1>Header Content</h1>将会显示在<slot name="header"></slot>位置,<p>Footer Content</p>将会显示在<slot name="footer"></slot>位置,而默认插槽中的内容仍会显示在<slot></slot>位置。

作用域插槽

作用域插槽是一种特殊类型的插槽,允许我们在父组件中访问子组件的数据。这在需要动态渲染内容时特别有用。

//子组件(MyComponent.vue)
<template>
  <div class="my-component">
    <slot :data="someData"></slot>
  </div>
</template>

<script>
export default {
  data() {
    return {
      someData: 'Hello from MyComponent!'
    };
  }
};
</script>
//父组件
<template>
  <MyComponent v-slot:default="slotProps">
    <p>{{ slotProps.data }}</p>
  </MyComponent>
</template>

在这个例子中,slotProps.data将会是'Hello from MyComponent!',并且会显示在父组件的<p>标签内。

八、事件总线(Event Bus)

事件总线的概念

事件总线是一个设计模式,它充当一个中间人,负责监听各个组件发布的事件,并分发给订阅这些事件的其他组件。在Vue中,事件总线通常是一个新的Vue实例,它提供了$emit$on$off方法,分别用于触发事件、监听事件和移除事件监听。

  • $emit:用于触发事件,并传递相关数据。
  • $on:用于监听事件,并定义事件触发时的回调函数。
  • $off:用于移除事件监听,防止内存泄漏。
  • 使用方式

    1. 创建一个空的Vue实例作为事件总线。
    2. 组件通过事件总线监听和触发事件来传递数据。
事件总线的实现步骤
  1. 创建事件总线
    • 你可以在一个单独的.js文件中创建一个新的Vue实例,并将其导出为事件总线。
    • 或者,你也可以在Vue项目的入口文件(如main.js)中,将事件总线挂载到Vue的原型上,使其成为全局可用的。
  2. 在组件中引入事件总线
    • 使用import语句将事件总线引入到你需要使用它的组件中。
  3. 触发和监听事件
    • 使用$emit方法在组件中触发事件,并传递相关数据。
    • 使用$on方法在另一个组件中监听这个事件,并定义回调函数来处理接收到的数据。
  4. 移除事件监听
    • 在组件销毁之前,使用$off方法移除所有不再需要的事件监听,以防止内存泄漏。
事件总线的例子

假设我们有两个兄弟组件ComponentAComponentB,它们需要通过事件总线进行通信。

1. 创建事件总线(event-bus.js):
import Vue from 'vue';
export const EventBus = new Vue();
2. 在组件中引入事件总线并使用

ComponentA.vue

<template>
  <button @click="sendMessage">Send Message to ComponentB</button>
</template>

<script>
import { EventBus } from '../event-bus.js';

export default {
  methods: {
    sendMessage() {
      EventBus.$emit('message-from-a', 'Hello from Component A!');
    }
  }
};
</script>

ComponentB.vue

<template>
  <div>
    <p>Message from ComponentA: {{ message }}</p>
  </div>
</template>

<script>
import { EventBus } from '../event-bus.js';

export default {
  data() {
    return {
      message: ''
    };
  },
  created() {
    EventBus.$on('message-from-a', (msg) => {
      this.message = msg;
    });
  },
  beforeDestroy() {
    EventBus.$off('message-from-a');
  }
};
</script>

在这个例子中,当ComponentA中的按钮被点击时,它会通过事件总线触发一个名为message-from-a的事件,并传递一条消息。然后,ComponentB会监听这个事件,并在接收到消息时更新其数据。最后,在ComponentB销毁之前,它会移除对message-from-a事件的监听,以防止内存泄漏。

事件总线的优缺点
优点
  • 灵活性:事件总线允许组件之间进行灵活的通信,不受组件层级结构的限制。
  • 解耦性:通过事件总线进行通信的组件之间不需要直接引用或依赖彼此,降低了组件之间的耦合度。
缺点
  • 调试困难:随着应用程序规模的增大,事件总线中的事件和监听器可能会变得非常复杂和难以管理。
  • 内存泄漏风险:如果忘记在组件销毁前移除事件监听器,可能会导致内存泄漏。

因此,在使用事件总线时,建议制定明确的事件命名规范,并在组件销毁前及时移除不再需要的事件监听器。此外,对于大型或复杂的应用程序,可以考虑使用更高级的状态管理解决方案(如本篇第六个 Vuex)来替代事件总线。

码字不易,字字皆心血。在此,诚挚地请求各位友友们动动手指,给予一个点赞,让这份努力得到认可与鼓励。友友们的每一次点赞,都是对我莫大的支持与激励!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/2237441.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

“穿梭于容器之间:C++ STL迭代器的艺术之旅”

引言&#xff1a; 迭代器&#xff08;Iterator&#xff09;是C STL&#xff08;标准模板库&#xff09;中非常重要的一部分&#xff0c;它提供了一种统一的方式来遍历容器中的元素。无论容器是数组、链表、树还是其他数据结构&#xff0c;迭代器都能够以一致的方式访问这些数据…

el-scrollbar 动态更新内容 鼠标滚轮无效

有以下功能逻辑&#xff0c;实现了一个时间轴组件&#xff0c;点击、-号后像地图那样放大组件以显示不同的UI。 默认显示年月&#xff1a; 当点击一下加号时切换为年&#xff1a; 当点击减号时切换为日&#xff1a; 即加号、减号点击就是在年月日显示进行切换。给Scrollvie…

Linux【基础篇】

-- 原生罪 linux的入门安装学习 什么是操作系统&#xff1f; 用户通过操作系统和计算机硬件联系使用。桥梁~ 什么是Linux&#xff1f; 他是一套开放源代码&#xff08;在互联网上找到Linux系统的源代码&#xff0c;C语言写出的软件&#xff09;&#xff0c;可以自由 传播&…

C++类(5)

1.<<和>>操作符重载 我们该如何重载操作符<<和>>呢&#xff1f; 如果在类里面&#xff0c; void operator<<(ostream& out) {out << _year << "年" << _month << "月" << _day <&l…

【MM-Align】学习基于输运的最优对齐动力学,快速准确地推断缺失模态序列

代码地址 - > github传送 abstract 现有的多模态任务主要针对完整的输入模态设置&#xff0c;即每个模态在训练集和测试集中要么是完整的&#xff0c;要么是完全缺失的。然而&#xff0c;随机缺失的情况仍然没有得到充分的研究。在本文中&#xff0c;我们提出了一种新的方…

高精度算法-保姆级讲解

目录 1.什么是高精度算法 2.高精度加法 3.高精度减法 4.高精度乘法 5.高精度除法 &#xff08;高精度除以低精度&#xff09; 6.高精度阶乘&#xff08;n个低精度数相乘&#xff09; 1.什么是高精度算法 高精度算法&#xff08;High Accuracy Algorithm&#xff09;是…

vue大疆建图航拍功能实现

介绍 无人机在规划一块区域的时候&#xff0c;我们需要手动的给予一些参数来影响无人机飞行&#xff0c;对于一块地表&#xff0c;无人机每隔N秒在空中间隔的拍照地表的一块区域&#xff0c;在整个任务执行结束后&#xff0c;拍到的所有区域照片能够完整的表达出一块地表&…

learnopencv系列三:GrabCut和DeepLabv3分割模型在文档扫描应用中的实现

文章目录 一、使用OpenCV实现自动文档扫描1.1 图片预处理1.2 查找轮廓1.3 检测角点1.4 仿射变换1.5 Streamlit Web App1.5.1 设置扫描函数和图像下载链接函数1.5.2 streamlit app1.5.3 测试结果 二&#xff1a;DeepLabv3文档分割2.1 项目背景2.2 合成数据集2.2.1 图像收集与预处…

SQLite的BLOB数据类型与C++二进制存储学习记录

一、BLOB数据类型简介 Blob&#xff08;Binary Large Object&#xff09;是一种用于存储二进制数据的数据类型&#xff0c;在数据库中常用于存储图片、音频和视频等大型&#xff08;大数据量&#xff09;的二进制数据[1-2]。需要注意的是&#xff0c;SQLite中BLOB类型的单对象最…

C# 自己编写web服务

文件后缀响应 "text/html"; 文件后缀响应 "application/json"; httpListenerContext.Response.ContentType 文件后缀响应; httpListenerContext.Response.AppendHeader("Access-Control-Allow-Origin", "*"); // L…

微服务day04

网关 网关路由 快速入门 创建新模块&#xff1a;hm-gateway继承hmall父项目。 引入依赖&#xff1a;引入网关依赖和nacos负载均衡的依赖 <?xml version"1.0" encoding"UTF-8"?> <project xmlns"http://maven.apache.org/POM/4.0.0"…

Agent框架调研:19种Agent架构对比分析

代理&#xff08;Agent&#xff09;指能自主感知环境并采取行动实现目标的智能体&#xff0c;即AI作为一个人或一个组织的代表&#xff0c;进行某种特定行为和交易&#xff0c;降低一个人或组织的工作复杂程度&#xff0c;减少工作量和沟通成本。 背景 目前&#xff0c;我们在…

ODOO学习笔记(4):Odoo与SAP的主要区别是什么?

Odoo 和 SAP 都是知名的企业资源规划&#xff08;ERP&#xff09;软件&#xff0c;它们之间存在以下一些主要区别&#xff1a; Odoo与SAP的区别 一、功能特点 功能广度 Odoo&#xff1a;提供了一整套全面的业务应用程序&#xff0c;涵盖了销售、采购、库存管理、生产、会计、…

python之正则表达式总结

正则表达式 对于正则表达式的学习&#xff0c;我整理了网上的一些资料&#xff0c;希望可以帮助到各位&#xff01;&#xff01;&#xff01; 我们可以使用正则表达式来定义字符串的匹配模式&#xff0c;即如何检查一个字符串是否有跟某种模式匹配的部分或者从一个字符串中将与…

【日志框架整合】Slf4j、Log4j、Log4j2、Logback配置模板

文章目录 一、日志框架介绍1、浅谈与slfj4、log4j、logback的关系2、性能方面3、Slf4j使用方法 二、log4j配置三、log4j2配置1、SpringBoot整合Log4j22、非SpringBoot项目引入的依赖3、log4j2-spring.xml文件&#xff08;Spring项目&#xff09;或log4j2.xml&#xff08;非Spri…

StarUML建模工具安装学习与汉化最新零基础详细教程【一键式下载】(适用于Windows、MacOS系统、Linux系统)

StarUML破解安装下载教程 前言&#xff1a; StarUML破解与汉化安装下载教程&#xff0c;仅供学习研究和交流使用&#xff0c;禁止作为商业用途或其他非法用途&#xff01; 仓库作者&#xff1a;X1a0He&#xff0c;经仓库作者授权使用。 目录 StarUML破解安装下载教程1. 下载…

【网络安全】2.3 安全的网络设计_2.防御深度原则

文章目录 一、网络架构二、网络设备三、网络策略四、处理网络安全事件五、实例学习&#xff1a;安全的网络设计结论 网络设计是网络安全的基础&#xff0c;一个好的网络设计可以有效的防止攻击者的入侵。在本篇文章中&#xff0c;我们将详细介绍如何设计一个安全的网络&#…

IoTDB 与 HBase 对比详解:架构、功能与性能

五大方向&#xff0c;洞悉 IoTDB 与 HBase 的详尽对比&#xff01; 在物联网&#xff08;IoT&#xff09;领域&#xff0c;数据的采集、存储和分析是确保系统高效运行和决策准确的重要环节。随着物联网设备数量的增加和数据量的爆炸式增长&#xff0c;开发者和决策者们需要选择…

如何找到系统中bert-base-uncased默认安装位置

问题&#xff1a; 服务器中无法连接huggingface&#xff0c;故需要自己将模型文件上传 ubuntu 可以按照这个链接下载 Bert下载和使用&#xff08;以bert-base-uncased为例&#xff09; - 会自愈的哈士奇 - 博客园 里面提供了giehub里面的链接 GitHub - google-research/be…

Qt 学习第十六天:文件和事件

一、创建widget对象&#xff08;文件&#xff09; 二、设计ui界面 放一个label标签上去&#xff0c;设置成box就可以显示边框了 三、新建Mylabel类 四、提升ui界面的label标签为Mylabel 五、修改mylabel.h&#xff0c;mylabel.cpp #ifndef MYLABEL_H #define MYLABEL_H#incl…