Vue组件通信原理剖析(三)provide/inject原理分析

首先我们先从一个面试题入手。

面试官问: “Vue中组件通信的常用方式有哪些?”
我答:
1. props
2. 自定义事件
3. eventbus
4. vuex
5. 还有常见的边界情况$parent、$children、$root、$refs、provide/inject
6. 此外还有一些非props特性$attrs、$listeners

面试官追问:“那你能分别说说他们的原理吗?”
我:[一脸懵逼]😳

在介绍provide和inject之前我们先简单看看其他几个常用属性。

如果要看别的属性原理请移步到Vue组件通信原理剖析(一)事件总线的基石 on和on和onemit和Vue组件通信原理剖析(二)全局状态管理Vuex

$parent / $root

解决问题:具有相同父类或者相同根元素的组件

// parant 
<child1></child1> 
<child2></child2>// child1
this.$parent.$on('foo', handle)// child2
this.$parent.$meit('foo')

$children

解决问题:父组件访问子组件实现父子通信

// parent
this.$children[0].childMethod = '父组件调用子组件方法的输出'

注意:$children是不能保证子元素的顺序

$attrs/$listeners

$attrs 包含了父作用域中不作为prop被识别且获取的特性绑定属性(class/style除外),如果子组件没声明prop,则包含除clas、style外的所有属性,并且在子组件中可以通过v-bind="$attrs"传入内部组件

// parent
<child foo="foo"></child>// child
<p>{{ $attrs.foo }}</p>

$listeners
包含了父作用域中的 (不含.native修饰器的)v-on事件监听器。它可以通过v-on="$listeners"传入内部组件在创建更高层次的组件时非常有用。
简单点讲它是一个对象,里面包含了作用在这个组件上所有的监听器(监听事件),可以通过v-on="$listeners"将事件监听指向这个组件内的子元素(包括内部的子组件)。
为了查看方便,我们设置`inheritAttrs: true,后面补充一下inheritAttrs。

// parent
<child @click="onclick"></child>// child 
// $listeners会被展开并监听
<p v-on="$listeners"></p>

$refs

解决问题:父组件访问子组件实现父子通信,和$children类似

// parent
<child ref="children"></child>mounted() {this.$refs.children.childMethod = '父组件调用子组件的输出'
}

provide/inject

解决问题:能够实现祖先和后代之间的传值

// ancestor
provide() {return {foo: 'foo'}
}// descendent
inject: ['foo']

那么问题来了,这个数据通信是什么样的机制呢?
我们先来看一个列子

// parent 父类
<template><div class=""><p>我是父类</p><child></child></div>
</template>export default {components: {child: () => import('./child')},provide: {foo: '我是祖先类定义provide'},
}// child 子类
<template><div class=""><p>我是子类</p><p>这是inject获取的值: {{ childFoo }}</p><grand></grand></div>
</template>
export default {components: {grand: () => import('./grand')},inject: { childFoo: { from: 'foo' } },
}// grand 孙类
<template><div class=""><p>我是孙类</p><p>这是inject获取的值: {{ grandFoo }}</p></div>
</template>
export default {components: {},inject: { grandFoo: { from: 'foo' } },
}

下面我结合上面的示例和源码一步一步分析一下:

  1. 先说说provide是怎么定义参数的,源码走起

    // 初始化Provide的实现
    export function initProvide (vm: Component) {const provide = vm.$options.provideif (provide) {vm._provided = typeof provide === 'function'? provide.call(vm): provide}
    }// vm.$options是怎么来的,是通过mergeOpitions得到的
    if (options && options._isComponent) {// optimize internal component instantiation// since dynamic options merging is pretty slow, and none of the// internal component options needs special treatment.initInternalComponent(vm, options);
    } else {vm.$options = mergeOptions(resolveConstructorOptions(vm.constructor),options || {},vm);
    }// 我们在看看mergeOptions的实现
    const options = {}
    let key
    for (key in parent) {mergeField(key)
    }
    for (key in child) {if (!hasOwn(parent, key)) {mergeField(key)}
    }
    function mergeField (key) {const strat = strats[key] || defaultStratoptions[key] = strat(parent[key], child[key], vm, key)
    }
    return options// 找到strat方法的实现
    strats.provide = mergeDataOrFn;export function mergeDataOrFn (parentVal: any,childVal: any,vm?: Component
    ): ?Function {if (!vm) {// in a Vue.extend merge, both should be functionsif (!childVal) {return parentVal}if (!parentVal) {return childVal}return function mergedDataFn () {return mergeData(typeof childVal === 'function' ? childVal.call(this, this) : childVal,typeof parentVal === 'function' ? parentVal.call(this, this) : parentVal)}} else {return function mergedInstanceDataFn () {// instance mergeconst instanceData = typeof childVal === 'function'? childVal.call(vm, vm): childValconst defaultData = typeof parentVal === 'function'? parentVal.call(vm, vm): parentValif (instanceData) {return mergeData(instanceData, defaultData)} else {return defaultData}}}
    }

    从上面的逻辑可以看出,在组件初始化时,会将vm.$options.provide这个函数赋值给provide,并把调用该函数得到的结果赋值给vm._provided,那么就会得到vm._provided = { foo: "我是祖先类定义provide" }

  2. 不要停,我们继续探究一下子孙组件中的inject是怎么实现的,上源码

    // 首先,初始化inject
    export function initInjections (vm: Component) {const result = resolveInject(vm.$options.inject, vm)if (result) {toggleObserving(false)Object.keys(result).forEach(key => {/* istanbul ignore else */if (process.env.NODE_ENV !== 'production') {defineReactive(vm, key, result[key], () => {warn(`Avoid mutating an injected value directly since the changes will be ` +`overwritten whenever the provided component re-renders. ` +`injection being mutated: "${key}"`,vm)})} else {defineReactive(vm, key, result[key])}})toggleObserving(true)}
    }// 初始化的inject实际上是resolveInject的结果,下面我们看看resolve都有哪些操作
    // 第一步:获取组件中定义的inject的key值,然后进行遍历
    // 第二步:根据key值获取对应的在provide中定义的provideKey,就比如上面的根据"childFoo"获取到"foo"
    // 第三步:通过source = source.$parent逐级往上循环在_provided中查找对应的provideKey
    // 第四步:如果找到,将实际的key值作为键,source._provided[provideKey]作为值,存为一个对象,当作这个函数的结果
    export function resolveInject (inject: any, vm: Component): ?Object {if (inject) {// inject is :any because flow is not smart enough to figure out cachedconst result = Object.create(null)const keys = hasSymbol? Reflect.ownKeys(inject): Object.keys(inject)for (let i = 0; i < keys.length; i++) {const key = keys[i]// #6574 in case the inject object is observed...if (key === '__ob__') continueconst provideKey = inject[key].fromlet source = vmwhile (source) {if (source._provided && hasOwn(source._provided, provideKey)) {result[key] = source._provided[provideKey]break}source = source.$parent}if (!source) {if ('default' in inject[key]) {const provideDefault = inject[key].defaultresult[key] = typeof provideDefault === 'function'? provideDefault.call(vm): provideDefault} else if (process.env.NODE_ENV !== 'production') {warn(`Injection "${key}" not found`, vm)}}}return result}
    }
    

说到这里,我们应该知道了provide/inject之间的调用逻辑了吧。最后,我们在用一句话总结一下:

当祖先组件在初始化时,vue首先会通过mergeOptions方法将组件中provide配置项合并vm.$options中,并通过mergeDataOrFn将provide的值放入当前实例的_provided中,此时当子孙组件在初始化时,也会通过合并的options解析出当前组件所定义的inject,并通过网上逐级遍历查找的方式,在祖先实例的-provided中找到对应的value值

至此,关于Vue的组件通信原理就介绍完了,希望能对大家有帮助。

全部文章链接

Vue组件通信原理剖析(一)事件总线的基石 on和on和onemit
Vue组件通信原理剖析(二)全局状态管理Vuex
Vue组件通信原理剖析(三)provide/inject原理分析

最后喜欢我的小伙伴也可以通过关注公众号“剑指大前端”,或者扫描下方二维码联系到我,进行经验交流和分享,同时我也会定期分享一些大前端干货,让我们的开发从此不迷路。
在这里插入图片描述

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

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

相关文章

iMX6开发板-uboot-网络设置和测试

本文章基于迅为IMX6开发板 将iMX6开发板通过网线连接到路由器&#xff0c;同时连接好调试串口&#xff0c;上电立即按 enter&#xff0c;即可进入 uboot。然后输入命令 pri&#xff0c;查看开发板当前的配置&#xff0c;如下图所示可以看到 ip 地址、子网掩码 等信息。 本文档测…

Django ajax 检测用户名是否已被注册

添加一个 register.html 页面 <!DOCTYPE html> <html lang"en"> <head><meta charset"UTF-8"><title>Title</title> </head> <body> <form><p>用户名<input id"username" type&…

pyqt5控件

背景色设置 self.tab.setStyleSheet("background: rgb(238, 233, 233)") self.but_0.setStyleSheet("background: rgb(0, 255, 255)")样式&#xff1a; self.but_0.setStyle(QStyleFactory.create("Windows"))字体&#xff1a; self.lineEdit.se…

详解JDBC连接数据库

一、概念 1. 为了能让程序操作数据库&#xff0c;对数据库中的表进行操作&#xff0c;每一种数据库都会提供一套连接和操作该数据库的驱动&#xff0c;而且每种数据库的驱动都各不相同&#xff0c;例如mysql数据库使用mysql驱动&#xff0c;oracle数据库使用oracle驱动&#xf…

ASP.NET MVC 自定义模型绑定1 - 自动把以英文逗号分隔的 ID 字符串绑定成 Listint...

直接贴代码了&#xff1a; CommaSeparatedModelBinder.cs using System; using System.Collections; using System.Collections.Generic; using System.Linq; using System.Reflection; using System.Web.Mvc;namespace MvcSample.Extensions {public class CommaSeparatedMode…

ZOJ4024 Peak

题意 给出一个数组 判断这个数组是否形成了一个“山峰” 即中间有个数最大 从第一个数到这个数递增 从这个数到最后一个数递减 模拟 从两端分别以递增和递减判断 看第一个不满足递增或递减的数是否相等并且没越界就可以了 AC代码&#xff1a; 1 #include<bits/stdc.h>2 u…

基本数据类型与String之间的转换

字符串转基本数据类型 调用基本数据类型对应的包装类中的方法parseXXX(String)或valueOf(String)即可返回相应基本类型。 基本数据类型转字符串 一种方法是将基本数据类型与空字符串&#xff08;""&#xff09;连接&#xff08;&#xff09;即可获得其所对应的字符串…

springmvc跨域问题

1、跨域问题&#xff1a; 按照网上所有的方法试了一遍&#xff0c;都没跨过去&#xff0c;正在无助之际&#xff0c;使用filter按照下面的方法解决的时候出现了转机&#xff1a; 添加filter&#xff1a; package com.thc.bpm.filter;import javax.servlet.*; import javax.serv…

柳传志给年轻人的建议:比起过日子,更要奔日子

改革开放的 40 年&#xff0c;是柳传志实现人生价值的 40 年。 十一届三中全会后&#xff0c;伴随“科学的春天”&#xff0c;迎着改革开放的大潮&#xff0c;柳传志“下海”了。但他并没想到&#xff0c;自己选择的电脑行业&#xff0c;让他和联想集团站在了潮头。 从 1984 年…

成功秀了一波scala spark ML逻辑斯蒂回归

1、直接上官方代码&#xff0c;调整过的&#xff0c;方可使用 package com.test import org.apache.spark.{SparkConf, SparkContext} import org.apache.spark.mllib.classification.{LogisticRegressionModel, LogisticRegressionWithLBFGS} import org.apache.spark.mllib.e…

记录一次查询log的经历

一大早发现生产数据库的基础资料被删除。 由于每天都做了差异备份&#xff0c;而且是基础资料&#xff0c;这样数据就不会担心找不回来。 首先通过每天的差异本分文件进行查看数据丢失的大概时间&#xff0c;查到数据丢失是在17晚上备份过后18丢失的。 然后找18号的数据库执行记…

移动端轮播图

1. 页面布局 1.1 页面框架 <body><div class"box"><div class"tupian"><img src"4.webp" alt""><img src"1.webp" alt""><img src"2.webp" alt""><…

Boost 序列化

原文链接&#xff1a; https://blog.csdn.net/qq2399431200/article/details/45621921 1. 编译器 gcc, boost 1.55 2.1第一个简单的例子 —— Hello World &#xff0c;将字符串内容归档到文本文件中 #include <iostream>#include <fstream>#include <string>…

docker CE 的安装

一、Docker CE的安装1.先决条件运行环境&#xff1a;Ubuntu 64位或者其他支持Docker的64位系统运行配置&#xff0c;linux内核版本必须大于 3.10&#xff0c;否则会因为缺少容器运行所需的功能而出错。 2.在ubuntu下安装Docker CEUbuntu版本 Cosmic 18.10  Bionic 18.04 (…

nodeJS中的异步编程

nodejs 不是单线程 在博客项目中关于异步问题&#xff1a; 1.当用户添加一条博客时 需要通过post方式向服务器发送数据 后台获取用户以post方式拿到传送过来的数据 然后存入数据库&#xff1a; 上面的代码&#xff1a;创建一个空字符串 当用户向服务器发送请求时出发data事件将…

day01笔记

linux基本命令的学习&#xff1a; 1.查看主机名hostname 2.修改主机名hostnamectl set-hostname s16ds 3.linux命令提示符 [roots16ds ~]# # 超级用户的身份提示符 $ 普通用户的身份提示符4.修改命令提示符 PS1变量控制 [roots16ds ~]# echo $PS1 [\u\h \W]\$PS1[\u\h \w \t]…

angular 路由

1. vscode编辑器快速新建主路由&#xff1a; ng-router注意修改为 根路由为&#xff1a;‘forRoot()’app-route.module.ts;{ path:,redirectTo:/login,pathMatch:full } 当路由为空的时候&#xff0c;会重定向到/login路由&#xff0c;必须加上pathMatch:full 1 import { Rou…

nodeJs 操作数据库

首先在node中下载mysql包 npm install mysql 连接数据库 var mysql require(mysql); var con mysql.createConnection({host : localhost,user : root,password : root,database : blog });开启链接 con.connect();执行增删改查 不同功能创建不同的sql语句即可…

shell字体颜色应用

输出特效格式控制&#xff1a; \033[0m 关闭所有属性 \033[1m 设置高亮度 \03[4m 下划线 \033[5m 闪烁 \033[7m 反显 \033[8m 消隐 \033[30m -- \033[37m 设置前景色 \033[40m -- \033[47m 设置背景色 光标位置等的格式控制&#xff1a; …

Spring Boot 统一结果封装

ResultVo, 返回结果对象 Data public class ResultVo<T> {private Integer code;private String message;private T data; }ResultVoUtil, 封装返回结果 public class ResultVoUtil {public static<T> ResultVo<T> sucess(T data) {ResultVo<T> result…