vue组件传值方式有哪些

news2025/1/18 14:00:42


Vue 作为一个轻量级的前端框架,核心两大特性就是响应式编程和组件化。

本文针对组件之间传值做详细讲解。 Vue就是由一个一个的组件构成的,组件化是它的精髓,也是最强大的功能之一。而组件实例的作用域是相互独立的,这就意味着不同组件之间的数据无法相互引用。

但在实际项目开发过程中,我们需要访问其他组件的数据,这样就就有了组件通信的问题。在 vue 中组件之间的关系有:父子,兄弟,隔代。针对不同的关系,怎么实现数据传递,下面展开说明。

如上图所示:
父子关系:A与B,A与C,B与D,C与E
兄弟关系:B与C
隔代关系(可能隔更多代):A与D,A与E
跨级关系:B与E,D与E等

一、父组件向子组件传值  props

<!--父组件页面-->
<template>
    <div id="app">
        <!--子组件-->
        <child-box :message="params"></child-box> //把params的值传给子组件
    </div>
</template>

<script>
// 引入子组件
import ChildBox from './childBox.vue'

export default {
    // 初始化子组件
    components: {
        ChildBox
    },
    data() {
        return {
            params: 'hello' // 如果params值为 undefined 则会使用default的值
        }
    }
}
</script>
<!--子组件页面-->
<template>
    <div class="child-box">{{ message }}</div>
</template>

<script>
    export default {
        /**
         * props功能:让组件接受外部传过来的数据
         * 
         * 1.传递数据 <child-box :message="hello"></child-box>
         * 
         * 2.接收数据
         *   第一种方式(只接收)
         *   props: ['message']
         * 
         *   第二种方式(限制类型)
         *   props: { message: String }  多个 props: { message: String, name: '王新焱'}
         *   
         *   第三种方式(限制类型、限制必要性、指定默认值)  //这种方式更为严谨
         *   props: {
         *      message: {
         *          type: String,
         *          required: true,
         *          default: '王新焱'
         *      }
         *   }
         * 
         *  备注:props是只读的,Vue底层会监测你对props的修改,如果进行了修改,就会发出警告,若业务需求确实需要修改,那么请复制props的内容到data中一份,然后去修改data中的数据
         *  
         *  如果要使用默认值 default message值设置为 undefined即可 ->示例:<child-box :message="undefined"></child-box>
         * 
         */
        props: {
            message: {
               type: String, // 指定数据类型
               required: true, // 指定数据为必填项
               default: '王新焱' // 指定默认值,如果message绑定的值设置为 undefined 即会使用该值
            }
        }
    }
</script>

总结:

1.父组件中注册子组件,在子组件标签中添加子组件props中创建的属性,把需要传给子组件的值赋给该属性。

2.子组件在props中创建一个属性,用以接收父组件传过来的值。

3.父子组件的关系可以总结为prop向下传递,事件向上传递。

4.父组件通过prop给子组件下发数据,子组件通过事件给父组件发送信息,这就是单向数据流的表现形式。

二、子组件向父组件传值  $emit

<!-- 父组件页面 -->
<template>
    <div id="app">
        <!-- 引入子组件,定义一个v-on的方法监听子组件的状态-->
        <child-box :message="params" @childFnGetParent="parentFn"></child-box>

        <P>{{ msg }}</P>
    </div>
</template>

<script>
// 引入子组件
import ChildBox from './childBox.vue'

export default {
    // 初始化子组件
    components: {
        ChildBox
    },
    data() {
        return {
            params: 'hello',
            msg: ''
        }
    },
    methods: {
        parentFn(val) {
            // val 就是子组件传过来的值
            this.msg = val
        }
    }
}
</script>
<!-- 子组件页面 -->
<template>
    <div class="child">
        <button @click="parentFn">子组件向父组件传值</button>
    </div>
</template>

<script>
    export default {
        data() {
            return {
                msg: '我是来自子组件的消息'
            }
        },
        methods: {
            parentFn() {
                // childFnGetParent 是在父组件v-on监听的方法,第二个参数this.msg是需要传递的值
                this.$emit('childFnGetParent', this.msg)
            }
        }
    }
</script>

总结

1.子组件中需要以某种方式例如点击事件的方法来触发一个自定义事件

2.将需要传的值作为$emit的第二个参数,该值将作为实参传给响应自定义事件的方法

3.在父组件中注册子组件并在子组件标签上绑定对自定义事件的监听

4.在通信中,无论是子组件向父组件传值还是父组件向子组件传值,他们都有一个共同点就是有中间介质,子向父的介质是自定义事件,父向子的介质是props中的属性。理解这两点对于父子通信就好理解了

子组件:通过$emit()方法发布事件广播

父组件:捕获到子组件向外触发的事件,然后可执行相应的方法

三、非父子组件传值  EventBus

非父子组件最常用的是EventBus方案进行数据传递,定义方式有三种

/**
 * 方法一 抽离成一个单独的 js 文件 EventBus.js ,然后在需要的地方引入
 * EventBus.js
 */
import Vue from "vue"
export default new Vue()

/**
 * 方法二 直接挂载到全局
 * main.js
 */
import Vue from "vue"
Vue.prototype.$bus = new Vue()

/**
 * 方法三 注入到 Vue 根对象上
 * main.js
 */
import Vue from "vue"
new Vue({
    el:"#app",
    data:{
        Bus: new Vue()
    }
})

本文案例以第一种方法 新建 EventBus 展开讲述

1.新建公共文件 EventBus.js

/**
 * EventBus.js
 * 
 * eventBus 又称为事件总线,在vue中可以使用它来作为沟通桥梁的概念, 就像是所有组件共用相同的事件中心
 * 
 * 不管是父子组件,兄弟组件,跨层级组件等都可以使用它完成通信操作
 */

import Vue from 'vue'

// 创建一个 EventBus.js 文件,暴露一个 vue 实例
export default new Vue()

2.新建一个A组件

<!--组件A-->
<template>
    <div class="child-A">
        <span>年龄:{{ value }}</span>
        <button @click="getValue">修改年龄</button>
    </div>
</template>

<script>
import Bus from './EventBus.js'
export default {
    data() {
        return {
            value: 30
        }
    },
    methods: {
        getValue() {
            // 在A组件中通过$emit向外部发送自定义事件 (事件广播)
            Bus.$emit('getVal', this.value)
        }
    }
}
</script>

3.新建一个B组件

<!--组件B-->
<template>
    <div class="child-B">
        <span>年龄:{{ age }}</span>
        <button @click="getData">点击触发</button>
    </div>
</template>

<script>
import Bus from './EventBus.js'
export default {
    data() {
        return {
            age: 18
        }
    },
    mounted() {
        // B组件 用$on事件来接收外部事件
        Bus.$on('getVal', (data)=> {
            this.age = data // data为广播过来值
        })
    },
    methods: {
        getData() {
            this.age++
        }
    }
}
</script>

使用 EventBus 有一个弊端就是事件广播,这种方式不会自动销毁,所以避免回调函数重复执行,需要在destroyed生命周期中销毁广播事件

destroyed() {
    Bus.$off('eventName')  // 对Bus取消事件监听后 内存得到了释放
}

总结

$off() 会取消所有的事件订阅

$off('事件名') 会取消指定事件名的

$off('事件名', 回调) 会取消指定事件名的,指定回调

四、多层父子组件通信 (依赖注入)  provide / inject

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

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

provide/inject:简单来说就是在父组件中通过provider来提供变量,然后在子组件中通过inject来注入变量,不管组件层级有多深,只要调用了inject 那么就可以注入provide中的数据。

<!--父组件页面-->
<template>
    <div class="box">
        <!-- 引入子组件A-->
        <child-a></child-a>

        <!-- 引入子组件B-->
        <child-b></child-b>
    </div>
</template>

<script>
export default {
    provide: {
        name: '张三丰', // 将变量name提供给它的所有子组件及后代组件
        reload: this.reload // 也可以是一个函数
    },
    data() {
        return {
            isShow: false
        }
    },
    methods: {
        reload() {
            this.isShow = true
        }
    }
}
</script>
<!--组件A-->
<template>
    <div class="child-A">
        <div>这是组件A</div>

        <!-- 输出张三丰 -->
        <div>{{ name }}</div>
    </div>
</template>

<script>

export default {
    inject: ['name'], // 注入了从父组件中提供的name变量

    mounted() {
        console.log(this.name) // 张三丰
    }
}
</script>
<!--组件B-->
<template>
    <div class="child-B">
        <div>这是组件B</div>
        
        <!-- 输出张三丰 -->
        <div>{{ name }}</div>
    </div>
</template>

<script>
export default {
    inject: ['name'], // 注入了从父组件中提供的name变量

    mounted() {
        console.log(this.name) // 张三丰
    }
}
</script>

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

五、通过访问组件实例的方式  ref

父组件通过 ref获取子组件的实例 可以直接访问子组件里面的方法和属性

<!--父组件页面-->
<template>
    <div class="box">
        <!-- 引入子组件A-->
        <child-a ref="childA"></child-a>

        <button @click="getChildData">获取子组件的属性和方法</button>
    </div>
</template>

<script>
// 引入子组件
import ChildA from './childBox.vue'

export default {
    // 初始化子组件
    components: {
        ChildA
    },
    methods: {
        getChildData() {
            // 获取子组件的属性
            console.log(this.$refs.childA.txt) //我是子组件A

            // 获取子组件的方法
            console.log(this.$refs.childA.getName()) //张三丰
        }
    }
}
</script>
<!--组件A-->
<template>
    <div class="child-A">
        <div>这是组件A</div>
    </div>
</template>

<script>

export default {
    data() {
        return {
            txt: '我是子组件A',
            name: '张三丰'
        }
    },
    methods: {
        getName () {
            console.log(this.name)
        } 
    }
}
</script>

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

六、$children / $parent

$children:获取到一个组件实例,包含所有子组件(不包含孙子组件)的 VueComponent 对象数组,可以直接拿到子组件中所有数据和方法等

$parent:获取到一个组件实例,包含父节点的 VueComponent 对象,同样包含父节点中所有数据和方法

$children 方法讲解

<!--父组件页面-->
<template>
    <div class="box">
        <!-- 引入子组件A-->
        <child-a ref="childA"></child-a>

        <!-- 引入子组件A-->
        <child-b></child-b>

        <button @click="getChildData">获取子组件的属性和方法</button>
    </div>
</template>

<script>
// 引入子组件
import ChildA from './childBox.vue'
import ChildB from './childBox2.vue'

export default {
    // 初始化子组件
    components: {
        ChildA,
        ChildB
    },
    methods: {
        getChildData() {
            // 获取第一个组件的name
            console.log(this.$children[0].name) //张三丰
            
            // 获取第一个组件getTitle方法
            console.log(this.$children[0].getTitle()) //我是子组件A
            
            // 获取第二个组件的name
            console.log(this.$children[1].name) //张无忌
        }
    }
}
</script>
<!--组件A-->
<template>
    <div class="child-A">
        <div>这是组件A</div>
    </div>
</template>

<script>

export default {
    data() {
        return {
            txt: '我是子组件A',
            name: '张三丰'
        }
    },
    methods: {
        getTitle () {
            console.log(this.txt)
        } 
    }
}
</script>
<template>
    <div class="child-B">
        <div>这是组件B</div>
    </div>
</template>

<script>
export default {
    data() {
        return {
            txt: '我是子组件B',
            name: '张无忌'
        }
    }
}
</script>

$parent 方法讲解

<!--父组件页面-->
<template>
    <div class="box">
        <!-- 引入子组件A-->
        <child-a ref="childA"></child-a>
    </div>
</template>

<script>
// 引入子组件
import ChildA from './childBox.vue'

export default {
    // 初始化子组件
    components: {
        ChildA
    },
    data() {
        return {
            title: '这是父页面',
            name: '父组件',
            age: 20
        }
    },
    methods: {
        getAge() {
            console.log(this.age)
        }
    }
}
</script>
<!--组件A-->
<template>
    <div class="child-A">
        <div>这是组件A</div>

        <button @click="getParentData">A组件获取父组件的属性和方法</button>
    </div>
</template>

<script>

export default {
    methods: {
        getParentData() {
            console.log(this.$parent.title) //获取父页面title属性
            console.log(this.$parent.getAge()) //执行父页面getAge方法
        }
    }
}
</script>

七、slot 插槽传值

子组件的数据通过插槽的方式传给父组件使用,要显示内容由父组件决定

<!--父组件页面-->
<template>
    <div class="box">
        <!--引入子组件A-->
        <child-a ref="childA" v-slot="slotProps">
            <!--父组件使用slot插槽方式获取子组件数据-->  
            {{ slotProps.user.name }}
            
            {{ name }}
        </child-a>
    </div>
</template>

<script>
// 引入子组件
import ChildA from './childBox.vue'

export default {
    // 初始化子组件
    components: {
        ChildA
    },
    data() {
        return {
            name: '张三丰'
        }
    }
}
</script>
<!--组件A-->
<template>
    <div class="child-A">
        <!-- <div>这是组件A</div> -->
        <slot :user="user"></slot>
    </div>
</template>

<script>

export default {
    data() {
        return {
            user: {
                name: '张无忌'
            }
        }
    }
}
</script>

以上几种也是vue组件传值主流的方式了,根据业务的不同场景选择不同的方式,此外vuex也是组件传值最常用的方式,我计划把这个知识点单独列出来进行讲解,后续会附上链接。

友情赠送一张小图

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

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

相关文章

priority_queue 接口使用(仿函数、函数指针解决优先级队列存放自定义类型元素、指针类型元素)

一、priority_queue 优先队列本质就是 堆 堆&#xff1a; 完全二叉树&#xff0c;任意结点比其孩子结点小->小根堆, 任意结点比其孩子结点大->大根堆, 头文件包含&#xff1a;#include<queue> 二、优先级队列的模板参数列表&#xff1a; template <class …

钓鱼邮件有哪些特征?

钓鱼邮件以其极具迷惑的特性&#xff0c;让很多人都上当受骗。一般的钓鱼邮件都有哪些特征呢&#xff1f; 1、一般像邮件中带有链接很可能是钓鱼邮件&#xff0c;收到不明邮件不要打开&#xff0c;更不能点击不明链接。 2、邮件中携带可疑附件&#xff0c;并且邮件正文具有引导…

Unity——新输入系统Input System

1.安装 安装&#xff1a; 直接到包管理器Window > Package Manager安装即可&#xff0c;安装后提示需要重启&#xff0c;重启后即可使用。 注意&#xff1a; 在Project Settings中的Player设置里将Active Input Handling设置为Input System。 需要将默认场景中的EventSyst…

线程安全☞有序性

什么是有序性&#xff1f; 在开发中&#xff0c;我们通常按照从上到下的顺序编写程序指令&#xff0c;并且希望cpu和编译器按照我们预先编写的顺序去执。但往往cpu和编译器为了提高性能、优化指令的执行顺序&#xff0c;会将我们编写好的程序指令进行重排序。 此时如果是在单…

如何将as代码上传到Gitee

一、Git的安装 1.这个直接到官网上下载&#xff0c;不多说了。Git - Downloads (git-scm.com) 2.在AS中添加Git的位置 二、AS中Gitee插件的安装 ⒈点击File中的Settings 2.安装插件 三.Gitee中账号的注册以及相关操作 1.注册账号 2.新建仓库 3. 在插件中注册Gitee账号 注…

声音经济@2023: 「夹缝」求生,「希望」不远

【潮汐商业评论/ 原创】“明明很困&#xff0c;但怎么也睡不着了。”Andy如是说。每每此刻&#xff0c;Andy便轻车熟路地拿起身旁的手机&#xff0c;随机打开音频APP&#xff0c;或是广播剧&#xff0c;或是播客节目&#xff0c;亦或是音乐和相声。伴随着声音的“催眠”&#x…

Cadence PCB仿真使用Allegro PCB SI元器件模型配置DevicesModels使用默认参数配置方法图文教程

⏪《上一篇》   🏡《总目录》   ⏩《下一篇》 目录 1,概述2,配置方法3,总结1,概述 本文简单介绍使用Allegro PCB SI元器件模型配置DevicesModels使用默认参数配置的方法。 2,配置方法 第1步:打开待仿真的PCB文件,并确认软件为Allegro PCB SI 如果,打开软件不是A…

将微信小程序转换uniapp进行迁移的步骤以及遇到的问题总结

目录 前言 一、迁移步骤 第一步&#xff1a;安装miniprogram-to-uniapp 插件 第二步&#xff1a;查看是否安装成功 第三步&#xff1a;使用插件进行转换 第四步&#xff1a;使用hbuilder X运行转换后的项目并在微信小程序编辑器查看 第五步&#xff1a;调试修改 二、处…

黑马学ElasticSearch(九)

目录&#xff1a; &#xff08;1&#xff09;数据聚合-聚合的分类 &#xff08;2&#xff09;数据聚合-DSL实现Bucket聚合 &#xff08;3&#xff09; 数据聚合-DSL实现Metrics聚合 &#xff08;4&#xff09;数据聚合-RestClient实现聚合 &#xff08;5&#xff09;数据聚…

重复生成数据itertools.repeat()

【小白从小学Python、C、Java】【计算机等级考试500强双证书】 【Python-数据分析】 重复生成数据 itertools.repeat() [太阳]选择题 以下python代码结果错误的一项是? import itertools print("【执行】xitertools.repeat(abc,3)") xitertools.repeat(abc,3) print…

xml中xsd/xsi/xsl/xmlns的区别和联系

&#xff08;2023.01.13Fri才明白&#xff0c;具体解析xml文件的不是xml文件本身&#xff0c;xsi也不是在解析xml文件&#xff0c;只是检验xml文件是否符合标准&#xff0c;xsd是在限制xml文件的内容。而具体解析某个element究竟是什么样式、起到什么作用&#xff0c;是需要接收…

可信AI年度观察 | 生成式AI技术及产品快速迭代,生成内容质量及安全性仍需提升

自2021年起&#xff0c;生成式人工智能&#xff08;以下简称“生成式AI”&#xff09;连续两年入选Gartner《Hype Cycle for Artificial Intelligence》&#xff0c;被认为是未来重要的AI技术趋势。2022年以来&#xff0c;生成式AI产品不断涌现&#xff0c;生成内容模态多样&am…

Python机器学习:认识机器学习

&#x1f315; 机器学习是什么&#xff1f; ⭐️&#xff1a;机器学习是一门涉及数学、统计学、计算机科学等的学科。 ⭐️&#xff1a;机器学习是从一个大量的已知数据中&#xff0c;学习如何对未知的新数据进行预测&#xff0c;并且可以随着学习内容的增加&#xff0c;提高对…

让你彻彻底底理解“并发”以及“并行”

1. 前言 我们做后端的人&#xff0c;一般会遇到别人给你说高并发&#xff0c;那我们知道什么叫并发吗&#xff1f;&#xff1f;&#xff1f; 什么叫并行吗&#xff1f;&#xff1f;&#xff1f; 今天这篇文章的目的其实就是为了讲明白什么叫并发&#xff0c;并行。好了废话不多…

二维数组八个方向的判断--牛客刷题1023-病菌感染

铁子和顺溜上生物课的时候不小心将几滴超级病菌滴到了培养皿上&#xff0c;这可急坏了他们。 培养皿可以被看成一个n*n的方格&#xff0c;最初病菌滴在了这n*n的格子中的某些格子&#xff0c;病菌的传染方式是这样的&#xff0c;如果一个方格与两个或多个被感染的方格相邻(两个…

2023,数字政务潮水已至

城市是时代发展最鲜活的物理存在&#xff0c;政务是城市的“中枢系统”&#xff0c;将数字技术广泛应用于政府管理服务&#xff0c;既能提高政府公共服务的履约效能&#xff0c;亦顺应经济社会数字化转型的大趋势。 作者|子雨 出品|产业家 从集聚资源求增长&#xff0c;到…

Linux的文件系统和软硬链接

目录 文件系统 磁盘的物理结构&#xff1a; 磁盘的存储结构 磁盘的逻辑抽象结构 管理文件 那么文件如何与属于自己的内容关联起来&#xff1f; 创建一个文件&#xff0c;OS做了什么&#xff1f; 删除一个文件&#xff0c;OS做了什么&#xff1f; 软硬链接 硬链接 软连…

Python学习笔记——字典

字典是一种映射类型 &#xff0c;它是一个无序的集合。内部元素是键值对形式出现&#xff0c;即一个关键字&#xff08;key&#xff09;与一个值&#xff08;value&#xff09;的组合&#xff08;“键&#xff0d;值”对&#xff09;。字典是一种可变容器模型&#xff0c;且可存…

视频在线点播功能如何实现?

1. 视频点播需求分析1.1 需求描述视频点播需求如下&#xff1a;1、学生可以在windows浏览器上在线观看视频。2、播放器具有快进、快退、暂停等基本功能。1.2 视频点播解决方案1.2.1 流媒体详细参考&#xff1a;https://baike.baidu.com/item/%E6%B5%81%E5%AA%92%E4%BD%93/98740…

2023年网络安全比赛--网络安全应急响应中职组(超详细)

一、竞赛时间 180分钟 共计3小时 二、竞赛阶段 竞赛阶段 任务阶段 竞赛任务 竞赛时间 分值 1.找出被黑客修改的系统别名,并将倒数第二个别名作为Flag值提交; 2.找出系统中被植入的后门用户删除掉,并将后门用户的账号作为Flag值提交(多个用户名之间以英文逗号分割,如:admi…