面试官:Vue 中父子组件通信有哪些方式?

自己先想一分钟。

无可否认,现在无论大厂还是小厂都已经用上了 Vue.js 框架,简单易上手不说,教程详尽,社区活跃,第三方套件还多。真的是前端开发人员必备技能。而且在面试当中也往往会问到关于 Vue 方面的各种问题,其中大部分面试官会问到如上这种问题。

最近一直在做 Vue项目代码层面上的优化,说实话,优化别人的代码真是件痛苦的事情,功能实现尚且不说,就说代码规范我就能再写出一篇文章来。真的是无规范不成方圆,规范这个东西太重要了!有点扯了,回到主题,咳咳,那就谈谈我对上面的面试题的理解吧,文笔有限,不妥之处,欢迎在文章结尾留言斧正啊,正啊,啊!

概述

几种通信方式无外乎以下几种:

Prop(常用)

$emit (组件封装用的较多)

.sync语法糖 (较少)

$attrs 和 $listeners (组件封装用的较多)

provide 和 inject (高阶组件/组件库用的较多)

其他方式通信

详述

下面逐个介绍,大神请绕行。

1. Prop

英式发音:[prɒp]。这个在我们日常开发当中用到的非常多。简单来说,我们可以通过 Prop 向子组件传递数据。用一个形象的比喻来说,父子组件之间的数据传递相当于自上而下的下水管子,只能从上往下流,不能逆流。这也正是 Vue 的设计理念之单向数据流。而 Prop 正是管道与管道之间的一个衔接口,这样水(数据)才能往下流。说这么多,看代码:

// Js

let Child = Vue.extend({

template: '

{{ content }}

',

props: {

content: {

type: String,

default: () => { return 'from child' }

}

}

})

new Vue({

el: '#app',

data: {

message: 'from parent'

},

components: {

Child

}

})

你可以狠狠的戳这里查看Demo!浏览器输出:

from parent

2. $emit

英式发音:[iˈmɪt]。官方说法是触发当前实例上的事件。附加参数都会传给监听器回调。按照我的理解不知道能不能给大家说明白,先简单看下代码吧:

let MyButton = Vue.extend({

template: 'click',

data () {

return {

greeting: 'vue.js!'

}

},

methods: {

triggerClick () {

this.$emit('greet', this.greeting)

}

}

})

new Vue({

el: '#app',

components: {

MyButton

},

methods: {

sayHi (val) {

alert('Hi, ' + val) // 'Hi, vue.js!'

}

}

})

你可以狠狠的戳这里查看Demo! 大致逻辑是酱婶儿的:当我在页面上点击按钮时,触发了组件 MyButton 上的监听事件 greet,并且把参数传给了回调函数 sayHi 。说白了,当我们从子组件 Emit(派发) 一个事件之前,其内部都提前在事件队列中 On(监听)了这个事件及其监听回调。其实相当于下面这种写法:

vm.$on('greet', function sayHi (val) {

console.log('Hi, ' + val)

})

vm.$emit('greet', 'vue.js')

// => "Hi, vue.js"

3. .sync 修饰符

这个家伙在 vue@1.x 的时候曾作为双向绑定功能存在,即子组件可以修改父组件中的值。因为它违反了单向数据流的设计理念,所以在 vue@2.0 的时候被干掉了。但是在 vue@2.3.0+ 以上版本又重新引入了这个 .sync 修饰符。但是这次它只是作为一个编译时的语法糖存在。它会被扩展为一个自动更新父组件属性的 v-on 监听器。说白了就是让我们手动进行更新父组件中的值了,从而使数据改动来源更加的明显。下面引入自官方的一段话:

在有些情况下,我们可能需要对一个 prop 进行“双向绑定”。不幸的是,真正的双向绑定会带来维护上的问题,因为子组件可以修改父组件,且在父组件和子组件都没有明显的改动来源。

既然作为一个语法糖,肯定是某种写法的简写形式,哪种写法呢,看代码:

v-bind:title="doc.title"

v-on:update:title="doc.title = $event">

于是我们可以用 .sync 语法糖简写成如下形式:

废话这么多,如何做到“双向绑定” 呢?让我们进段广告,广告之后更加精彩!

好的,欢迎回来。假如我们想实现这样一个效果:改变子组件文本框中的值同时改变父组件中的值。怎么做?列位不妨先想想。先看段代码:

{{ userName }}

let Login = Vue.extend({

template: `

姓名:

`,

props: ['name'],

data () {

return {

text: ''

}

},

watch: {

text (newVal) {

this.$emit('update:name', newVal)

}

}

})

new Vue({

el: '#app',

data: {

userName: ''

},

components: {

Login

}

})

你可以狠狠的戳这里查看Demo!下面划重点,代码里有这一句话:

this.$emit('update:name', newVal)

官方语法是:update:myPropName 其中 myPropName 表示要更新的 prop 值。当然如果你不用 .sync 语法糖使用上面的 .$emit 也能达到同样的效果。仅此而已!

官方语法是:update:myPropName 其中 myPropName 表示要更新的 prop 值。当然如果你不用 .sync 语法糖使用上面的 .$emit 也能达到同样的效果。仅此而已!

4. $attrs 和 $listeners

官网对 $attrs 的解释如下:

包含了父作用域中不作为 prop 被识别 (且获取) 的特性绑定 (class 和 style 除外)。当一个组件没有声明任何 prop 时,这里会包含所有父作用域的绑定 (class 和 style 除外),并且可以通过 v-bind="$attrs" 传入内部组件——在创建高级别的组件时非常有用。

官网对 $listeners 的解释如下:

包含了父作用域中的 (不含 .native 修饰器的) v-on 事件监听器。它可以通过 v-on="$listeners" 传入内部组件——在创建更高层次的组件时非常有用。

我觉得 $attrs 和 $listeners 属性像两个收纳箱,一个负责收纳属性,一个负责收纳事件,都是以对象的形式来保存数据。看下面的代码解释:

:foo="foo"

:bar="bar"

@one.native="triggerOne"

@two="triggerTwo">

从 Html 中可以看到,这里有俩属性和俩方法,区别是属性一个是 prop 声明,事件一个是 .native 修饰器。

let Child = Vue.extend({

template: '

{{ foo }}

',

props: ['foo'],

created () {

console.log(this.$attrs, this.$listeners)

// -> {bar: "parent bar"}

// -> {two: fn}

// 这里我们访问父组件中的 `triggerTwo` 方法

this.$listeners.two()

// -> 'two'

}

})

new Vue({

el: '#app',

data: {

foo: 'parent foo',

bar: 'parent bar'

},

components: {

Child

},

methods: {

triggerOne () {

alert('one')

},

triggerTwo () {

alert('two')

}

}

})

你可以狠狠的戳这里查看Demo! 可以看到,我们可以通过 $attrs 和 $listeners 进行数据传递,在需要的地方进行调用和处理,还是很方便的。当然,我们还可以通过 v-on="$listeners" 一级级的往下传递,子子孙孙无穷尽也!

一个插曲!

当我们在组件上赋予了一个非Prop 声明时,编译之后的代码会把这些个属性都当成原始属性对待,添加到 html 原生标签上,看上面的代码编译之后的样子:

parent foo

这样会很难看,同时也爆了某些东西。如何去掉?这正是 inheritAttrs 属性的用武之地!给组件加上这个属性就行了,一般是配合 $attrs 使用。看代码:

// 源码

let Child = Vue.extend({

...

inheritAttrs: false, // 默认是 true

...

})

再次编译:

parent foo

5. provide / inject

他俩是对CP, 感觉挺神秘的。来看下官方对 provide / inject 的描述:

provide 和 inject 主要为高阶插件/组件库提供用例。并不推荐直接用于应用程序代码中。并且这对选项需要一起使用,以允许一个祖先组件向其所有子孙后代注入一个依赖,不论组件层次有多深,并在起上下游关系成立的时间里始终生效。

看完描述有点懵懵懂懂!一句话总结就是:小时候你老爸什么东西都先帮你存着等你长大该娶媳妇儿了你要房子给你买要车给你买只要他有的尽量都会满足你。下面是这句话的代码解释:

let Son = Vue.extend({

template: '

son

',

inject: {

house: {

default: '没房'

},

car: {

default: '没车'

},

money: {

// 长大工作了虽然有点钱

// 仅供生活费,需要向父母要

default: '¥4500'

}

},

created () {

console.log(this.house, this.car, this.money)

// -> '房子', '车子', '¥10000'

}

})

new Vue({

el: '#app',

provide: {

house: '房子',

car: '车子',

money: '¥10000'

},

components: {

Son

}

})

你可以狠狠的戳这里查看Demo!

6. 其他方式通信

除了以上五种方式外,其实还有:

EventBus

思路就是声明一个全局Vue实例变量 EventBus , 把所有的通信数据,事件监听都存储到这个变量上。这样就达到在组件间数据共享了,有点类似于 Vuex。但这种方式只适用于极小的项目,复杂项目还是推荐 Vuex。下面是实现 EventBus 的简单代码:

// 全局变量

let EventBus = new Vue()

// 子组件

let Child = Vue.extend({

template: '

child

',

created () {

console.log(EventBus.message)

// -> 'hello'

EventBus.$emit('received', 'from child')

}

})

new Vue({

el: '#app',

components: {

Child

},

created () {

// 变量保存

EventBus.message = 'hello'

// 事件监听

EventBus.$on('received', function (val) {

console.log('received: '+ val)

// -> 'received: from child'

})

}

})

你可以狠狠的戳这里查看Demo!

Vuex

官方推荐的,Vuex 是一个专为 Vue.js 应用程序开发的状态管理模式。

$parent

父实例,如果当前实例有的话。通过访问父实例也能进行数据之间的交互,但极小情况下会直接修改父组件中的数据。

$root

当前组件树的根 Vue 实例。如果当前实例没有父实例,此实例将会是其自己。通过访问根组件也能进行数据之间的交互,但极小情况下会直接修改父组件中的数据。

broadcast / dispatch

他俩是 vue@1.0 中的方法,分别是事件广播 和 事件派发。虽然 vue@2.0 里面删掉了,但可以模拟这两个方法。可以借鉴 Element 实现。有时候还是非常有用的,比如我们在开发树形组件的时候等等。

总结

啰嗦了这么多,希望看到的同学或多或少有点收获吧。不对的地方还请留言指正,不胜感激。父子组件间的通信其实有很多种,就看你在哪些情况下去用。不同场景不同对待。前提是你要心中有数才行!通过大神之路还有很远,只要每天看看社区,看看文档,写写Demo,每天进步一点点,总会有收获的。俗话说,三人行则必有我师,希望更多志同道合的小伙伴能聚在一起交流技术!下面的群快满了,可以加我 Q1769617251 。备注 vue 即可。

html中的父子通信代码,Vue.js 父子组件通信的十种方式相关推荐

  1. Vue.js 父子组件通信的十种方式;告诉世界前端也能做 AI;你可能不知道的14个JavaScript调试技巧...

    记得点击文章末尾的"阅读原文"查看哟~ 下面先一起看下本周的摘要吧~ 想了解老用户如何参与阿里云双十一1折拼团特惠主机的,可以看第二条推送,文中提供了两种方法~,一起看看本周有哪些 ...

  2. vue的父子孙之间组件通信和兄弟之间的组件通信

    父子孙组件之间的组件通信 面试题:vue组件之间有哪些通信方式呢? 第一种:props+this.$emit: props: 父组件A通过v-bind绑定数据,传递数据给子组件,子组件用props接收 ...

  3. 360奇舞团钟恒:选用Vue.js进行组件化开发,我们遇到了哪些坑?

    责编:陈秋歌,关注前端开发领域,寻求报道或者投稿请发邮件chenqg#csdn.net. 欢迎加入"CSDN前端开发者"微信群,参与热点.难点技术交流.请加群主微信「Rachel_ ...

  4. Vue.js教程-组件化开发

    Vue.js教程-组件化开发 前言 Vue组件化 什么是组件化 Vue单页面开发的解释 组件化思想 组件的使用 原理 实际开发中的使用-父子组件 父子组件传递数据 父传子-props用法 子传父-th ...

  5. [译]使用Webpack提高Vue.js应用程序的4种方式

    [译]使用Webpack提高Vue.js应用程序的4种方式 原文地址 ​ Webpack是开发Vue.js单页应用程序的重要工具.通过管理复杂的构建步骤,您可以更轻松地开发工作流程,并优化应用程序的大 ...

  6. Vue.js 定义组件模板的七种方式

    转载自  Vue.js 定义组件模板的七种方式 在 Vue 中定义一个组件模板,至少有七种不同的方式(或许还有其它我不知道的方式): 字符串 模板字面量 x-template 内联模板 render ...

  7. 两个读书笔记:springboot+vue.js分布式组件全栈开发训练营 + 大数据开发基础

    (springboot+vue.js分布式组件全栈开发训练营原文在notion中, 大数据开发在思维导图中, 这个博客只是保存, 无法阅读. ) what is different between s ...

  8. vue组件传值的十种方式

    vue组件传值的十种方式 一.props 父传子 子组件 // 第一种数组方式 props: [xxx, xxx, xxx] // 第二种对象方式 props: { xxx: Number, xxx: ...

  9. vue在html中执行js代码,Vue.js 和 Vue.runtime.js

    Vue官方中文文档: Vue有两个版本: 完整版:vue.js.vue.min.js(运行时版+编译器)(编译器:将模板字符串编译成为JS渲染函数的代码) 运行时版:vue.runtime.js.vu ...

最新文章

  1. android自定义文件选择,关于安卓自定义本地文件选择库的实现
  2. Angular InjectionToken的一个具体使用例子
  3. 「每天一道面试题」如何理解方法的重载与覆盖?
  4. Java面试题16 牛客 以下java程序代码,执行后的结果是()
  5. 《MySQL排错指南》——1.9 许可问题
  6. C/C++ scanf 函数中%s 和%c 的简单差别
  7. 松哥整理了 15 道 Spring Boot 高频面试题,看完当面霸
  8. dlibdotnet 人脸相似度源代码_使用dlib中的深度残差网络(ResNet)实现实时人脸识别 - supersayajin - 博客园...
  9. 如果企业级客户不爽,亚马逊AWS该怎么办?
  10. 【PostgreSQL-9.6.3】创建、修改、删除PostgreSQL数据库
  11. selenium 问题:OSError: [WinError 6] 句柄无效
  12. Linux6新建分区,centos6中添加一块新的硬盘并分区的方法介绍
  13. C#.NET身份证验证算法
  14. 视频剪辑的方法,视频裂变
  15. AngularJS - $uibModal - 自定义模态框大小
  16. eas账号是什么意思_账户和帐号是什么意思,有区别吗?
  17. [TeXPage] 打造好用的 LaTeX 在线编辑器
  18. apk私钥_Android应用apk的程序签名详解
  19. Java网课笔记整理
  20. AE开发 遇到未能加载文件或程序集的问题 FileNotFoundException

热门文章

  1. 【初试复试第一】脱产在家二战上岸——上交819考研经验
  2. 品牌台式计算机配置,办公电脑什么牌子好 办公台式电脑配置推荐
  3. freeswitch之sofia模块
  4. 监控摄像机选型攻略之技术类型选用
  5. 用一个程序员的角度来遐想第五代或第六代计算机
  6. linux安装配置opencv
  7. 面试车祸现场——面试题总结
  8. 磨金石教育摄影技能干货分享|古风人像修图与调色技巧
  9. Elasticsearch HTTP查询
  10. ubuntu设置网卡默认启动_ubuntu配置网卡的办法