做者:Milos Protic前端
译者:前端小智vue
来源:vuejsdevelopersgit
你知道的越多,你不知道的越多
github点赞再看,养成习惯
面试
本文 GitHub:github.com/qq449245884… 上已经收录,更多往期高赞文章的分类,也整理了不少个人文档,和教程资料。欢迎Star和完善,你们面试能够参照考点复习,但愿咱们一块儿有点东西。bash
有人说递归很难理解,也有人不这么认为。递归函数简单的定义是:一个自调用函数,这意味着它将在执行的某个时刻调用本身。微信
从理论上讲,递归是一种须要两个属性的行为:数据结构
我们没法决定哪个更重要。若是没有结束点,递归将成为一个无限循环,可是若是一组规则就不能实现指望的行为,因此二者都存在才能使它正常工做。app
在 Vue 中,递归很是有用。固然,不只仅在 Vue 中,我们能够遵循上面的规则在任何框架中实现递归行为。所以,根据给定的定义,我们能够说递归组件是调用自身的组件。框架
递归组件何时有用? 只要我们须要使用相同的模板结构,但须要使用分层输入数据,就可使用递归。 若是树状视图(用于显示文件夹结构),网站上的注释,嵌套菜单等组件等等。
接着,我们创建一个场景来演示递归组件的用途。
想象一下,咱像往常同样来上班,给本身冲杯咖啡,开始阅读咱最喜欢的博客。忽然,我们的老板来了,说须要实现一个新的页面,在这个页面上,显示全部的文件夹、子文件夹和文件,且文件结构数量不肯定。能够显示10
个、5
个或100
个文件夹。接着,我们喝着咖啡,开始挠头思考如何解决这个问题。最终,我们会想到使用递归遍从来实现。
解决这个问题的组件的最少数量是1
,但在我们的示例中,我们会建立两个组件:
root
组件folder
组件固然,我们首先搞点数据来用:
如前所述,当我们有分层组织的数据,其中子数据具备与其父数据相同的结构时,递归就派上用场了。
const root = {
text: 'Root Folder',
leaf: false,
expanded: true,
children: [{
text: 'Sub Folder 1',
leaf: false,
expanded: false,
children: [{
text: 'Sub Sub Folder 1',
leaf: false,
expanded: false,
children: [{
text: 'SomeFile1.js',
leaf: true
}]
}, {
text: 'Sub Sub Folder 2',
leaf: false,
expanded: false,
children: []
}, {
text: 'SomeFile.txt',
leaf: true
}]
}]
}
复制代码
有了上面的数据,我们开始建立组件。
这个组件是我们文件夹树的起点。它会开始全部子元素的沉浸,可是若是须要,它也能够显示一些独立的信息,由于它不是递归自己的一部分。
root
组件将包含一个folder
属性,我们会把root
数据对象绑定到该属性上。此属性将传递给子组件,子组件将递归地建立基于它的文件夹树结构。
Template
<template>
<ul class="folders">
<li>Folders</li>
<folder v-bind:folder="folder"></folder>
</ul>
</template>
复制代码
代码
import Folder from './Folder.vue';
export default {
name: 'root',
props: {
folder: Object
},
components: {
Folder
}
};
复制代码
样式
ul.folders {
padding: 1rem;
margin: 0;
box-sizing: border-box;
width: 100%;
list-style: none
}
ul.folders > li:first-child {
padding: 1rem 1rem 1rem 0
}
复制代码
就是这么简单。 ‘
此组件负责渲染树中的每一个文件夹。它负责显示关于当前文件夹的信息,并渲染其子文件夹(若是有的话)。此外,这些文件夹是可单击的,经过单击其中一个,组件将显示其子文件夹和文件。
Template
<template>
<li class="folder" v-bind:class="[folder.leaf ? 'is-leaf' : 'is-folder']">
<span v-on:click="expand()">{{ folder.text }}</span>
<ul class="sub-folders" v-if="folder.children && folder.children.length > 0" v-show="folder.expanded">
<folder v-for="child in folder.children" v-bind:folder="child"></folder>
</ul>
<div class="folder-empty" v-else v-show="!folder.leaf && folder.expanded">No Data</div>
</li>
</template>
复制代码
Code
export default {
name: "folder",
props: {
folder: Object
},
methods: {
expand() {
if (this.folder.leaf) {
return;
}
this.folder.expanded = !this.folder.expanded;
}
}
};
复制代码
样式:
li.is-folder {
padding: 1rem;
border-left: 1px solid #d3d3d3;
margin-bottom: 0.5rem
}
li.is-folder > span {
padding: 0.5rem;
border: 1px solid #d3d3d3;
cursor: pointer;
display:inline-block
}
li.is-leaf {
padding: 0 0 0 1rem;
color: #000;
}
ul.sub-folders {
padding: 1rem 1rem 0 0;
margin: 0;
box-sizing: border-box;
width: 100%;
list-style: none
}
div.folder-empty {
padding: 1rem 1rem 0 1rem;
color: #000;
opacity: 0.5
}
复制代码
为了使用我们刚刚建立的组件,所须要作的就是将root
组件导入到须要此功能的地方,并传递数据结构。例如,在 App.vue
中使用:
Template
<template>
<div class="vue-app">
<root v-bind:folder="root"></root>
</div>
</template>
复制代码
Code
import Root from './Root.vue';
export default {
name: 'app',
data: function () {
return {
root: {
text: 'Root Folder',
leaf: false,
expanded: true,
children: [{
text: 'Sub Folder 1',
leaf: false,
expanded: false,
children: [{
text: 'Sub Sub Folder 1',
leaf: false,
expanded: false,
children: [{
text: 'SomeFile1.js',
leaf: true
}]
}, {
text: 'Sub Sub Folder 2',
leaf: false,
expanded: false,
children: []
}, {
text: 'SomeFile.txt',
leaf: true
}]
}]
}
}
},
components: {
Root
}
};
复制代码
运行效果:
递归并不像看起来那么难,它只是用不一样的输入参数一次又一次地执行相同的代码块,直到达到结束点。但愿本文可以更好帮你们理解递归以及如何使用Vue建立递归组件。
编辑中可能存在的bug无法实时知道,过后为了解决这些bug,花了大量的时间进行log 调试,这边顺便给你们推荐一个好用的BUG监控工具 Fundebug。
干货系列文章汇总以下,以为不错点个Star,欢迎 加群 互相学习。
由于篇幅的限制,今天的分享只到这里。若是你们想了解更多的内容的话,能够去扫一扫每篇文章最下面的二维码,而后关注我们的微信公众号,了解更多的资讯和有价值的内容。