slot多作用域 vue_vue 深度长文之slot 篇

0303efc755d1da71001d4970e9581de0.png

今天我们将分析我们经常使用的 vue 功能 slot 是如何设计和实现的,本文将围绕 普通插槽 和 作用域插槽 以及 vue 2.6.x 版本的 v-slot 展开对该话题的讨论。当然还不懂用法的同学建议官网先看看相关 API 先。接下来,我们直接进入正文吧

普通插槽

首先我们看一个我们对于 slot 最常用的例子

be261fee5b03b06e20edd904e9ba742a.png

然后我们直接使用,页面则正常显示一下内容

e70ee38aeec454da750b15ac41d37fd8.png

然后,这个时候我们使用的时候,对 slot 内容进行覆盖

this is slot custom content.

内容则变成下图所示

2203ec66e9059e719251cc5c149ddebb.png

对于此,大家可能都能清楚的知道会是这种情况。今天我就将带领大家直接看看 vue 底层对 slot 插槽的具体实现。

vm.$slots

我们开始前,先看看 vue 的 Component 接口上对 $slots 属性的定义

$slots: { [key: string]: Array }; 

多的咱不说,咱直接 console 一下上面例子中的 $slots

456d478bc76b8000d224147dbb67326a.png

剩下的篇幅将讲解 slot 内容如何进行渲染以及如何转换成上图内容

renderSlot

看完了具体实例中 slot 渲染后的 vm.$slots 对象,这一小篇我们直接看看 renderSlot 这块的逻辑,首先我们先看看 renderSlot 函数的几个参数都有哪些

c428d11989ad9f3454aaada6c215b80b.png

这里我们先不看 scoped-slot 的逻辑,我们只看普通 slot 的逻辑。

const slotNodes = this.$slots[name]nodes = slotNodes || fallbackreturn nodes

这里直接先取值 this.$slots[name] ,若存在则直接返回其对其的 vnode 数组,否则返回 fallback。看到这,很多人可能不知道 this.$slots 在哪定义的。解释这个之前我们直接往后看另外一个方法

150d8dfca6dff8e3ba0e5b2964bb8c4f.png

看完 resolveSlots 的参数后我们接着往后过其中具体的逻辑。如果 children 参数不存在,直接返回一个空对象

const slots = {}if (!children) { return slots}

如果存在,则直接对 children 进行遍历操作

43a58441631980ef7207d563e6ca6b41.png

slots 获取到值后,则进行一些过滤操作,然后直接返回有用的 slots

4bd08f6c3b75f07715389e1a97a440c5.png

我们从上面已经知道了 vue 对 slots 是如何进行赋值保存数据的。而在 src/core/instance/render.js 的 initRender 方法中则是对 vm.$slots 进行了初始化的赋值。

23b1381e26f19c4772c3fee9153b3a98.png

了解了是 vm.$slots 这块逻辑后,肯定有人会问:你这不就只是拿到了一个对象么,怎么把其中的内容给搞出来呢?别急,我们接着就来讲一下对于 slot 这块 vue 是如何进行编译的。这里咱就把 slot generate 相关逻辑过上一过,话不多说,咱直接上代码

6089e8f3379286cf6ee8facdb1fc6e08.png

注:上面的 slotName 在 src/compiler/parser/index.js 的 processSlot() 函数中进行了赋值,并且 父组件编译阶段用到的 slotTarget 也在这里进行了处理

6d48e604cc1acbdf8ac988552ebe945b.png

随即在 genData() 中使用 slotTarget 进行 data 的数据拼接

if (el.slotTarget && !el.slotScope) { data += `slot:${el.slotTarget},`}

此时父组件将生成以下代码

13f73dff48a8f357a3701a1085176e86.png

然后当 el.tag 为 slot 的情况,则直接执行 genSlot()

else if (el.tag === 'slot') { return genSlot(el, state)}

按照我们举出的例子,则子组件最终会生成以下代码

619b54d47df7c73225c14cb9021396e9.png

作用域插槽

上面我们已经了解到 vue 对于普通的 slot 标签是如何进行处理和转换的。接下来我们来分析下作用域插槽的实现逻辑。

1、vm.$scopedSlots

了解之前还是老规矩,先看看 vue 的 Component 接口上对 $scopedSlots 属性的定义

$scopedSlots: { [key: string]: () => VNodeChildren };

其中的 VNodeChildren 定义如下:

declare type VNodeChildren = Array<?VNode | string | VNodeChildren> | string;

先来个相关的例子

2137c9ac472d3016f39770945027a5d6.png

然后进行使用

73b9c7d4a7239b86017d08c9dee50036.png

效果如下

10f30856628ab2e55d785a09338b682f.png

从使用层面我们能看出来,子组件的 slot 标签上绑定了一个 text 以及 :msg 属性。然后父组件在使用插槽使用了 slot-scope 属性去读取插槽带的属性对应的值

注:提及一下 processSlot() 对于 slot-scope 的处理逻辑

e99572f7b0287cce60455bee9fe2b575.png

从上面的代码我们能看出,vue 对于这块直接读取 slot-scope 属性并赋值给 AST 抽象语法树的 slotScope 属性上。而拥有 slotScope 属性的节点,会直接以 **插槽名称 name 为 key、本身为 value **的对象形式挂载在父节点的 scopedSlots 属性上

然后在 src/core/instance/render.js 的 renderMixin 方法中对 vm.$scopedSlots 则是进行了如下赋值:

if (_parentVnode) { vm.$scopedSlots = _parentVnode.data.scopedSlots || emptyObject}

然后 genData() 里会进行以下逻辑处理

if (el.scopedSlots) { data += `${genScopedSlots(el, el.scopedSlots, state)},`}

紧接着我们来看看 genScopedSlots 中的逻辑

6cfc42f9f6f1e5b55e5a18343d2ea0c9.png

然后我们再来看看 genScopedSlot 是如何生成 render function 字符串的

0773a7c1379e0058c2b5796bbbaa2d66.png

我们把上面例子的 $scopedSlots 打印一下,结果如下

2e7919b41a6b6d6d172f5a407af88208.png

然后上面例子中父组件最终会生成如下代码

c0e745661f3ef2d079a6c11945bc4cdf.png

renderSlot(slot-scope)

上面我们提及对于插槽 render 逻辑的时候忽略了 slot-scope 的相关逻辑,这里我们来看看这部分内容

138dee0cb128df624be63c2671f9b061.png

这里我们看看 renderHelps 里面的 _u ,即 resolveScopedSlots,其逻辑如下

53b7843e5d9300f3907b92775a6cbb7e.png

这块会对 attrs 和 v-bind 进行,对于这块内容上面我已经提过了,要看请往上翻阅。结合我们的例子,子组件则会生成以下代码

c8ef4a46f00bf82d6723307886ddc153.png

v-slot

1、基本用法

vue 2.6.x 已经出来有一段时间了,其中对于插槽这块则是放弃了 slot-scope 作用域插槽推荐写法,直接改成了 v-slot 指令形式的推荐写法(当然这只是个语法糖而已)。下面我们将仔细谈谈 v-slot 这块的内容。

在看具体实现逻辑前,我们先通过一个例子来先了解下其基本用法

0d8c39489a5c4f9be26aa5b37b767a92.png

然后进行使用

6f18b93bd853fb69274374825e23f1b0.png

页面展示效果如下

a0435b8ba2b52accafbcd5eb4e05feb5.png

相同与区别

接下来,咱来会会这个新特性

round 1. $slots & $scopedSlots

$slots 这块逻辑没变,还是沿用的以前的代码

// $slotsconst options = vm.$optionsconst parentVnode = vm.$vnode = options._parentVnodeconst renderContext = parentVnode && parentVnode.contextvm.$slots = resolveSlots(options._renderChildren, renderContext)

$scopedSlots 这块则进行了改造,执行了 normalizeScopedSlots() 并接收其返回值为 $scopedSlots 的值

f5242f3261c5cd4237f7b72ad0a73058.png

接着,我们来会一会 normalizeScopedSlots ,首先我们先看看它的几个参数

0ff4e1cb5683d987253eb22dd21de5d3.png
  • 首先,如果 slots 参数不存在,则直接返回一个空对象 {}
if (!slots) { res = {}}
  • 若 prevSlots 存在,且满足系列条件的情况,则直接返回 prevSlots
d5d6d9064b2a3a626d0ee59de2b265c4.png

注:这里的 $key , $hasNormal , $stable 是直接使用 vue 内部对 Object.defineProperty 封装好的 def() 方法进行赋值的

def(res, '$stable', isStable)def(res, '$key', key)def(res, '$hasNormal', hasNormalSlots)复制代码
  • 否则,则对 slots 对象进行遍历,操作 normalSlots ,赋值给 key 为 key,value 为 normalizeScopedSlot 返回的函数 的对象 res
ceda417cd49f0e451746e1e5f2b7d9fa.png
  • 随后再次对 normalSlots 进行遍历,若 normalSlots 中的 key 在 res 找不到对应的 key,则直接进行 proxyNormalSlot 代理操作,将 normalSlots 中的 slot 挂载到 res对象上
  • 接着,我们看看 normalizeScopedSlot() 都做了些什么事情。该方法接收三个参数,第一个参数为 normalSlots,第二个参数为 key,第三个参数为 fn
ae48db860cd00f21d96ca8a4fcf90ec7.png

参考文章:

https://juejin.im/post/5cced0096fb9a032426510ad

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

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

相关文章

android 扫描照片功能,Android自定义View- 雷达扫描图

首先来看看效果图&#xff1a;CSDN博客地址这里写图片描述这里我使用了两种实现方式&#xff1a;继承 view 实现。继承 surfaceview 实现。为什么会有两种实现方式呢&#xff1f;主要是因为我在继续加入一些自定义功能的时候&#xff0c;如果是继承 view &#xff0c;出现了卡顿…

SmartSVN for Mac 使用说明

文章目录连接 SVN 服务器签出&#xff08;Check Out&#xff09;导出&#xff08;Export&#xff09;打开工作副本&#xff08;Open Working Copy&#xff09;打开和管理项目移除工作副本&#xff08;Remove Working Copy&#xff09;项目及工作副本目录导航窗口查看和配置被忽…

bldc 原理 方波控制_【百问百答】ST 电机控制实战问答合辑 | 连载之一

点击下方链接可以直接观看电机直播及直播答疑电堂​wx18257eb0e8c82435.h5.xiaoe-tech.com电堂​wx18257eb0e8c82435.h5.xiaoe-tech.com本次实战问答只讨论同步电机&#xff0c;不对步进电机做特别的讨论&#xff0c;希望有助于大家进行电机开发或者是电机应用。Q1&#xff1a;…

android动画编辑软件,ALM视频动画编辑

ALM视频动画编辑app手机端中最为专业,强大的视频编辑工具,丰富的功能堪比pc级别,并且实用流畅不卡顿,操作简单明了,上手容易,视频效果出色,更多丰富素材内容使用&#xff01;下载ALM视频动画编辑app开始体验吧&#xff01;ALM视频动画编辑介绍ALM视频动画编辑神器为你提供非常高…

openhub_介绍OpenHub框架

openhub本文介绍OpenHub框架 -基于Apache Camel的新的开源集成解决方案。 本文回答了一些问题&#xff0c;为什么您应该关心另一个集成框架&#xff0c;强弱属性以及如何使用OpenHub启动新项目。 OpenHub框架是Apache Camel&#xff0c;但经过改进…… 当然&#xff0c;您只能…

spring 导出csv_Spring批处理CSV处理

spring 导出csv总览 我们将讨论的主题包括使用Spring Batch进行批处理的基本概念&#xff0c;以及如何将数据从CSV导入数据库。 0 – Spring Batch CSV处理示例应用程序 我们正在构建一个应用程序&#xff0c;演示用于处理CSV文件的Spring Batch的基础。 我们的演示应用程序将…

python强制结束函数_为什么Python没有 main 函数?终于有人给出了正确答案!

毫无疑问&#xff0c;Python中没有所谓的 main 入口函数&#xff0c;但在网上经常有文章提到“Python中的main函数”和“建议编写main函数”等。他们的目的可能是模仿真实的 main 函数&#xff0c;但是经常有很多人被误导&#xff08;或误解&#xff09;并编写非常繁琐的代码。…

MacOS 如何显示/隐藏文件

文章目录显示/隐藏文件快捷键修改“访达”属性修改文件隐藏属性设置特殊文件名实现隐藏使用命令设置文件隐藏属性显示/隐藏文件 快捷键 按下 Shift Command . 可以显示隐藏型的文件&#xff0c;再按下 Shift Command . 则不显示隐藏型的文件 修改“访达”属性 defaults…

不相关子查询的工作方式是_课题组工作|Nucleic Acids Research|基于表达密度谱的特征子空间分离及相关单细胞转录组分群新算法...

大家好&#xff01;为大家分享本课题组近期发表在Nucleic Acids Research的文章&#xff0c;题目为 “Entropy subspace separation-based clustering for noise reduction (ENCORE) of scRNA-seq data”&#xff0c;文章提出了一种基于表达密度谱的特征选择方法&#xff0c;能…

java 9 module_Java 9:欢迎来到Module World

java 9 moduleJava 9已于9月21日正式发布&#xff0c;Eclipse从Eclipse Oxygen.1a&#xff08;4.7.1a&#xff09;支持Java 9&#xff0c;让我们进入模块世界。 从此处下载Java 9&#xff0c;并将其添加到Eclipse Installed JRE中&#xff0c;如下所示 就是这样&#xff0c;…

Mac OS 通过配置窗口来连接远程主机

点击连接后&#xff0c;就跳到命令终端窗口中&#xff0c;输入远程主机 root 用户的登录密码&#xff0c;则完成连接。

温度补偿计算公式_一种工业用温度测量模块的设计与实现

一种工业用温度测量模块的设计与实现罗伯特侯0 引言温度是工业生产中最重要的参数之一&#xff0c;因此温度测量设备在工业领域不可或缺。热电偶是工业场合中应用最广泛的温度传感器,它的主要特点是测温范围宽,价格低&#xff0c;同时结构简单,坚固耐用。笔者采用高集成度、高精…

微博如何发订阅消息_微信订阅号或将大变天,微博8年前就这么干了...

近日&#xff0c;微信又偷偷的做了一个大胆的尝试&#xff0c;将公众号消息排序改版&#xff0c;不再按照一直以来的“时间轴展示”&#xff0c;而是学起了微博&#xff0c;变成了“智能排序”。微信的“阅读效率优化”排序不过可能是为了独树一帜&#xff0c;微信管这种排序方…

indesign如何画弧线_彩铅画入门教程,如何给独角兽设计一款好发型

戳这里 → 查看“爱蜜干货文章目录”本次综合训练的目的1&#xff0e;挖掘你的绘画感和想象力&#xff0c;彩色鬃毛色彩大家可以自由发挥哦&#xff01;2.练习彩铅的长线条&#xff0c;现在练习的长条还是比较简单的&#xff0c;下次综合训练我们还会练习更加复杂的3.彩铅这种画…

微型计算机作为载体的部件是,大工11秋《计算机应用基础》辅导资料二

计算机应用基础辅导资料二主题&#xff1a;计算机基础知识的辅导资料学习时间&#xff1a;2011年10月10日&#xff0d;10月16日内容&#xff1a;这周我们主要学习课件&#xff0e;&#xff0e;第二章计算机的基础知识&#xff0c;本章的学习要求及需要掌握的重点内容如下&#…

markdown如何设置图片大小_Gitee(码云)实现免费 Markdown 图床

“阅读本文大概需要 6 分钟前言Markdown是一种易于上手的轻量级标记语言&#xff0c;由于其目的在于注重文字内容而不是排版&#xff0c;目前很受大家欢迎&#xff0c;写完一篇文档可以直接复制到其他各大平台上&#xff0c;不用担心格式字体等混乱问题但是文章中如果引用了某个…

适合利用计算机模拟的是,计算机模拟在数学建模中的应用

计算机模拟在数学建模中的应用计算机模拟是按时间来划分的&#xff0c;因为计算机模拟实质上是系统随时间变化而变化的动态写照&#xff0c;以下是小编搜集整理的一篇探究计算机模拟在数学建模应用的论文范文&#xff0c;供大家阅读参考。【摘要】本文主要阐述了如何利用计算机…

噪音声压和声功率的区别_南昌汽车隔音,深入了解汽车噪音的来源、危害以及解决方案...

汽车噪音带来的危害&#xff1a;汽车噪音对人体健康的影响是多方面的。噪音作用于人的中枢神经系统&#xff0c;使人们大脑皮层的兴奋与抑制平衡失调&#xff0c;导致条件反射异常&#xff0c;使脑血管张力遭到损害。这些生理上的变化&#xff0c;在早期能够恢复原状&#xff0…

lua加密教程_我们相信加密! 教程

lua加密教程许多人认为加密是一个复杂的主题&#xff0c;这很难理解。 可以实现其某些方面&#xff0c;但是每个人都可以理解它在更高层次上的工作方式。 这就是我要处理的这篇文章。 用简单的术语解释它是如何工作的&#xff0c;然后使用一些代码。 是的&#xff0c;我们信任…

生产用计算机软件管理台账,计算机台账管理系统

计算机台账管理系统计算机台账管理系统是什么&#xff1f;什么是计算机台账管理系统&#xff1f;对于设备管理而言&#xff0c;设备台账是其重要的组成部分&#xff0c;计算机台账管理系统对设备的编号、适用规格、年限、使用部门等具体信息进行管理&#xff0c;方便设备资产的…