Vue3(五):组件通信详解(九种方法)

主要有九种方法,以下是详细解释及使用方法:

1.props

props实现父子间的通信,是使用频率最高的。

(1)父传子:属性值是非函数。

以Father.vue和Child.vue 为例。

父组件中,引入子组件并给子组件绑定一个car参数,

子组件中,使用defineProps([''])接收,在template中,可以直接使用car

(2)子传父:属性值是函数

其实子传父前提父亲先给子传递一个函数,子接收到了再传给父。

比如我要把儿子的toy传给父亲,那么就需要在父中先定义一个getToy函数,用:sendToy=''getToy"传给子,然后子再去调用sendToy。

2.自定义事件emit

1. 概述:

自定义事件常用于:子 => 父

2. 原生Dom事件、自定义事件

(1)原生Dom事件:

  - 事件名是特定的(click、dbdlick、change等等)

  - 事件对象`$event`: 是包含事件相关信息的对象(`pageX`、`pageY`、`target`、`keyCode`)

(2)自定义事件:

  - 事件名是任意名称

  - <strong style="color:red">事件对象`$event`: 是调用`emit`时所提供的数据,可以是任意类型!!!</strong >

  1. vue2中 @click是自定义事件,可以通过.native修饰符变为原生DOM事件。

  2. 在vue3中@click是原生DOM事件

  3. `$event是事件对象

3. 例如:

子组件传给父亲toy,就可以这样做:父组件自定义一个send-toy方法,然后子组件使用defineEmits接收,再用emit传给父亲,之后父亲利用send-toy的回调保存传输过来的toy

3.mitt

1.概述

与消息订阅与发布(`pubsub`)功能类似,可以实现任意组件间通信

就是只要任意两个组件能够摸到emitter,就能实现通信。

2.使用mitt

(1)安装mitt

npm i mitt

(2)新建文件:src\utils\emitter.ts

// 引入mitt 
import mitt from "mitt";// 创建emitter
const emitter = mitt()/*// 绑定事件emitter.on('abc',(value)=>{console.log('abc事件被触发',value)})emitter.on('xyz',(value)=>{console.log('xyz事件被触发',value)})setInterval(() => {// 触发事件emitter.emit('abc',666)emitter.emit('xyz',777)}, 1000);setTimeout(() => {// 清理事件emitter.all.clear()}, 3000); 
*/// 创建并暴露mitt
export default emitter

3.使用

 emit触发,on绑定。

比如弟弟child2想要哥哥child1的玩具 ,那child2要用emitter.on绑定事件,然后child1用emitter.emit触发事件。

(注意:这里最好用onUnmounted解绑一下事件,如果有一天这个组件被卸载了,但是他化存在这个关系,就会对内存不友好) 

4.v-model

1.本质

实现父↔子间通信

其实一般很少使用v-model,但是在UI组件库中会大量使用v-model.

2.使用

(1)v-model用在html标签上

v-model本质其实是value和input的结合使用 

<!-- v-model用在html标签上 --><!-- <input type="text" v-model="username"> --><input type="text" :value="username" @input="username = (<HTMLInputElement>$event.target).value"><!-- <HTMLInputElement>是说明$event.target是一个html中输入类型的元素 -->

 可以实现双向绑定

(2)v-model用在组件标签

 只需要在组件标签中v-model="username",然后引入UI组件库。

组件标签上的`v-model`的本质:`:moldeValue` + `update:modelValue`事件。

组件标签中: 

<!-- 组件标签上使用v-model指令 --><AtguiguInput v-model="userName"/><!-- 组件标签上v-model的本质 --><AtguiguInput :modelValue="userName" @update:model-value="userName = $event"/>

 UI组件库中:

<template><div class="box"><!--将接收的value值赋给input元素的value属性,目的是:为了呈现数据 --><!--给input元素绑定原生input事件,触发input事件时,进而触发update:model-value事件--><input type="text" :value="modelValue" @input="emit('update:model-value',$event.target.value)"></div></template><script setup lang="ts" name="AtguiguInput">// 接收propsdefineProps(['modelValue'])// 声明事件const emit = defineEmits(['update:model-value'])</script>

(3) 修改modelValue

 可以修改modelValue的话,就意味着可以在组件标签上使用多个v-model.

多个v-model 

5.$attrs

1. 概述

`$attrs`用于实现当前组件的父组件,向当前组件的子组件通信(祖孙通信)。

2. 具体说明

`$attrs`是一个对象,包含所有父组件传入的标签属性。

   >  注意:`$attrs`会自动排除`props`中声明的属性(可以认为声明过的 `props` 被子组件自己“消费”了)

3.使用

先来了解一下attrs 

首先父向子传递多个响应式数据,但是子只接受一个a,那么剩下的都去哪里了呢。其实在attrs中存着,我们可以用$attrs看到

(1)祖传孙: 

 假如上述父子传递时,子不接收呢?而是用v-bind传给孙组件呢?

那么在孙组件中就可以用defineProps接收,并呈现出来。

这样就实现了祖向孙通信 

(2)孙传祖

在祖里面定义有一个方法

并把方法也传给子

<Child :a="a" :b="b" :c="c" :d="d" v-bind="{x:100,y:200}" :updateA="updateA"/>

 而子中依旧是这样:

<GrandChild v-bind="$attrs"/>

那么孙中只要接收到了方法,就可以绑定一个点击事件,把自己的定义的数字6传给祖的value,实现孙传祖。

6.$refs、$parent 

1. 概述:

$refs`用于 :父→子

$parent`用于:子→父

2.原理如下

3.使用

(1)ref

 先来看一下用ref是怎么修改的。首先用ref给Child1打个标识c1,然后配置一个修改方法changeToy

之后在child1中配置difineExpose就可以了,这样父就能点击修改子的玩具。实现父到子通信。

// 把数据交给外部defineExpose({toy,book})

 同理如果父亲想要修改child2的数据也同理。

(2)$refs

在父组件中用如下写法可以获取所有子组件的实例对象 

<button @click="getAllChild($refs)">获取所有子组件的实例对象</button>

 如图所示:可以获取

那么就可以利用refs修改所有子组件共同想修改的内容,例如修改所有子组件的玩具数量。 

记得在所有子组件里面配置defineExpose

// 把数据交给外部defineExpose({toy,book})

 (3)$parent

通过$parent可以看到父组件的实例对象 。这样可以通过在子组件中使用$parent并配置方法,从而修改父组件数据。实现子向父通信。

千万不要忘了在父组件中配置defineExpose

// 向外部提供数据defineExpose({house})

7.provide、inject

1.概述

实现祖孙组件直接通信

2. 具体使用

   * 在祖先组件中通过`provide`配置向后代组件提供数据

   * 在后代组件中通过`inject`配置来声明接收数据

3.例如

(1)祖传孙

例如把爷爷有的钱和车子都传给孙,应该怎么做呢?

在父组件中先从vue中import引入provide ,而后使用provide向后代提供数据

provide('名字',值)    需要注意的是这个值后面不要.value,这样会丢失响应式

 

在后代组件中,先import引入inject,而后使用inject接收数据。

需要注意的是,inject('名字','默认值')这2个参数 一个是名字一个是默认值,如果不写默认值的话,上面的car.brand和car.price就会报错。

(2)孙传祖

比如孙子华爷爷的钱。在孙组件里面点击一下按钮就能修改爷爷的钱。

首先在祖组件里面定义一个方法,在provide里面传的是money的值和这个方法给孙组件

那么在孙组件里面可以把传过来的值和方法 利用解构赋值,然后再添加一个按钮绑定这个方法,并传数据。就实现了孙传祖。

8.pinia

详见上一节:Vue3(四):Pinia-CSDN博客

9.slot插槽

假如要实现这样一个效果:

  

1.默认插槽

slot里面是默认内容,如果没有要插进来的内容,就会显示默认内容

//父组件中:<Category title="今日热门游戏"><ul><li v-for="g in games" :key="g.id">{{ g.name }}</li></ul></Category>
//子组件中:<template><div class="item"><h3>{{ title }}</h3><!-- 默认插槽 --><slot></slot></div></template>

2.具名插槽

在子组件标签里面把需要传递的内容使用template标签包括,用v-slot命名,也可以用#命名。 子组件slot中用相应的name="xxx"

父组件中:<Category title="今日热门游戏"><template v-slot:s1><ul><li v-for="g in games" :key="g.id">{{ g.name }}</li></ul></template><template #s2><a href="">更多</a></template></Category>
子组件中:<template><div class="item"><h3>{{ title }}</h3><slot name="s1"></slot><slot name="s2"></slot></div></template>

3.作用域插槽

像下面这样,父组件拿games会报错,因为这时候父组件是拿不到子组件的games的,涉及到一个作用域的内容。 

对于作用域插槽的解释:可以这样理解 

语法及使用:

第一种是使用v-slot="params",意思是吧template里面的内容放在默认插槽里面

第二种是也可以使用命名的方式,因为slot有默认的命名,default,所以可以用#default

10.总结

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

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

相关文章

34、链表-合并K个升序链表

思路 1、直接全部放入集合中&#xff0c;然后排序&#xff0c;在进行构造节点返回 2、使用归并排序的方式&#xff0c;两两排序合并&#xff0c;最后合并大的。 3、第三中思路就比较巧妙了&#xff0c;可以使用小根堆&#xff0c;每次弹出堆顶&#xff0c;最小值&#xff0c…

【计算机网络】http协议的原理与应用,https是如何保证安全传输的

✨✨ 欢迎大家来到景天科技苑✨✨ &#x1f388;&#x1f388; 养成好习惯&#xff0c;先赞后看哦~&#x1f388;&#x1f388; &#x1f3c6; 作者简介&#xff1a;景天科技苑 &#x1f3c6;《头衔》&#xff1a;大厂架构师&#xff0c;华为云开发者社区专家博主&#xff0c;…

基于 RT-Thread 的 CMUX 串口多路复用的详细使用

一、CMUX 软件包的介绍 CMUX&#xff08;Connection Multiplexing &#xff09;&#xff0c;即连接&#xff08;串口&#xff09;多路复用&#xff0c;其功能主要在一个真实的物理通道上虚拟多个通道&#xff0c;每个虚拟通道上的连接和数据通讯可独立进行。  CMUX 软件包常用…

DRF ModelSerializer序列化类

ModelSerializer序列化类 【0】准备 模型表创建 from django.db import modelsclass Book(models.Model):name models.CharField(max_length64, verbose_name书名)price models.DecimalField(max_digits6, decimal_places2, verbose_name价格)publish models.ForeignKey(…

【C++打怪之路】-- C++开篇

&#x1f308; 个人主页&#xff1a;白子寰 &#x1f525; 分类专栏&#xff1a;C打怪之路&#xff0c;python从入门到精通&#xff0c;魔法指针&#xff0c;进阶C&#xff0c;C语言&#xff0c;C语言题集&#xff0c;C语言实现游戏&#x1f448; 希望得到您的订阅和支持~ &…

三元运算符可选链操作符

三元运算符 &#xff08; ?: &#xff09;又叫条件运算符&#xff0c;接受三个运算数&#xff1a;条件 ? 条件为真时要执行的表达式 : 条件为假时要执行的表达式 <script>var a truevar b false// 单个条件a ? console.log(1) : console.log(2); // 1// 多个条件…

【SpringBoot】Spring Boot 项目中整合 MyBatis 和 PageHelper

目录 前言 步骤 1: 添加依赖 步骤 2: 配置数据源和 MyBatis 步骤 3: 配置 PageHelper 步骤 4: 使用 PageHelper 进行分页查询 总结 前言 Spring Boot 与 MyBatis 的整合是 Java 开发中常见的需求&#xff0c;特别是在使用分页插件如 PageHelper 时。Page…

vue-cli2 与vue-cli3,vue2与vue3 初始化项目,本地vue项目,详细解析区别(2024-04-19)

目录 1、区别&#xff08;vue-cli2 与 vue-cli3 &#xff09; 2、例子1&#xff08;vue2项目&#xff09; 2.1 版本与命令行 2.2 项目本地截图 2.3 项目文件解析 &#xff08;1&#xff09;package.json 文件 &#xff08;2&#xff09;webpack.dev.conf.js文件 &#…

[大模型]Qwen-7B-hat Transformers 部署调用

Qwen-7B-hat Transformers 部署调用 环境准备 在autodl平台中租一个3090等24G显存的显卡机器&#xff0c;如下图所示镜像选择PyTorch–>2.0.0–>3.8(ubuntu20.04)–>11.8 接下来打开刚刚租用服务器的JupyterLab&#xff0c;并且打开其中的终端开始环境配置、模型下…

【JavaScript】异步函数(async和await详解)

异步函数 ES8 的 async/await 旨在解决利用异步结构组织代码的问题。为此&#xff0c;ECMAScript 对函数进行了扩展&#xff0c;为其增加了两个新关键字&#xff1a;async 和 await。 1. async async 关键字用于声明异步函数。这个关键字可以用在函数声明、函数表达式、箭头…

NLP自然语言处理_序章

开一个新篇章&#xff0c;立一个flag&#xff0c;用一段时间来学习一下NLP&#xff0c;涨涨见识。 准备以B站 机器学习算法到transformer神经网络模型应用视频作为入门&#xff0c;此分类专门用于记录学习过程中的知识点以备自用。 一、何为NLP自然语言处理&#xff1f; NLP…

查看linux的主机配置脚本

废话不说 直接上指令 curl -Lso- bench.sh | bash 等待后&#xff0c;结果如图&#xff1a; 使用后没有问题&#xff0c;看情况使用 出事概不负责 介意勿用&#xff01;&#xff01;&#xff01;

esxi配置虚拟机定时开机

有个测试机器希望每晚关机&#xff0c;第2天早8点自动开机&#xff0c;linux系统中配置了定时关机任务。开机任务在esxi上配置如下 [rootlocalhost:~] cat /var/spool/cron/crontabs/root #min hour day mon dow command 1 1 * * * /sbin/tmpwatch.py 1 * *…

web server apache tomcat11-09-JNDI Datasource

前言 整理这个官方翻译的系列&#xff0c;原因是网上大部分的 tomcat 版本比较旧&#xff0c;此版本为 v11 最新的版本。 开源项目 从零手写实现 tomcat minicat 别称【嗅虎】心有猛虎&#xff0c;轻嗅蔷薇。 系列文章 web server apache tomcat11-01-官方文档入门介绍 web…

变形记---容灾恢复(一),异常崩溃引发服务器丢档或无法正常运行

接着上篇文章 变形记---抽象接口,屎山烂代码如何改造成优质漂亮的代码 ,我一直想写一些对年轻人有帮助的文档来,刚好最近有空就零零碎碎写了一些,罗列了一些提纲然后改再删,花了一个礼拜的时间。 写这一系列的 “变形记”,也是因为最近我给M部门面试服务器主程序开发的职…

现代软件为什么要采用微服架构

现代软件采用微服务架构是为了解决传统单体架构在开发、部署和维护大型应用时面临的一系列问题。以下是采用微服务架构的主要优势&#xff1a; 1. **模块化和组件化**&#xff1a;微服务通过将应用拆分为一系列小型、松耦合的服务来提高模块化水平。每个服务都是围绕特定的业务…

RabbitMQ 各种通信模式的Python实现

一、RabbitMQ 原理 1、基本原理 RabbitMQ是流行的开源消息队列系统&#xff0c;用erlang语言开发。RabbitMQ是AMQP&#xff08;高级消息队列协议&#xff09;的标准实现。支持多种客户端&#xff0c;如&#xff1a;Python、Java、Javascript、C#、C/C,Go等&#xff0c;支持AJ…

音响扫频分析音频特性

把音响的的扬声器断开&#xff0c;接负载电阻&#xff0c;再连接到AP仪器端进行扫频测试&#xff0c;可得到频响曲线。

开源大模型 Llama 3

开源大模型Llama 3是一个在多个领域都展现出卓越性能的大模型。下面将为您介绍Llama 3的特性和一些简单的使用案例。 一、Llama 3介绍 Llama 3在Meta自制的两个24K GPU集群上进行预训练&#xff0c;使用了超过15T的公开数据&#xff0c;其中5%为非英文数据&#xff0c;涵盖30多…

刀片式服务器的作用有哪些?

刀片式服务器是是一种紧凑的独立服务器&#xff0c;将多个服务器功能集成在一个紧凑的机箱中&#xff0c;这种服务器设计可以使数据中心更加高效地管理硬件资源&#xff0c;提高服务器的可靠性和可扩展性&#xff0c;在多种场景中得到广泛的应用。 刀片式服务器的作用有哪些呢&…