递归组件的实现原理

最近在开发一个小模块,碰到了这么一个需求: 在一个页面中显示一级分类,当一级分类有子类的话,点开一级分类,展现二级分类,假设二级分类中的个别某一项有属于他的子分类的时候,点开他,继续显示它的子分类。上面所说的每级分类中,子分类数目可能是一项或者多项,数目不固定,具体看返回来的数据有多少。

我擦嘞,这个不就是递归么。。。。。。然而,在vue中,官方是不推荐直接操作dom树的,因为这样会非常损耗性能。。。。这样一来,直接操作dom树这个想法就只能丢弃了。。。这可如何是好呢?于是,我就开始度娘了。。。。然后度到在官网里有递归组件这么一回事,但也仅限于介绍,其它特么的什么也木有,但是说了最重要的一点,一定要有name属性,因为这个name属性指代的就是组件本身。。。然后我就接着找啊找啊,找到了一篇不错的博文,具体链接地址: https://www.cnblogs.com/ychl/p/6075106.html 。不错的一篇文章,不过个人感觉可以再细化点,于是经过博主同意,进行二次个人解读。 有朋友就纳闷了:你咋那么懒。。。。我想说,一切为了项目,这可不是个人业余学习时间,我也是现在有点时间,怕以后忘记了,赶紧记起来,避免以后掉链子。

我们上面说到了,我们的递归应该是按照数据来的。废话,你如果没有子类,你递归的毛线啊,所以,一切以数据为准。请看下面的一些列数据,来自于隔壁家的老黄的 文章 : https://www.cnblogs.com/ychl/p/6075106.htmlvuejs学习--递归组件(树型表格分享)


vardata = [{
                "id": "1",
                "data": {
                    "menuName": "项目管理",
                    "menuCode": "",
                },
                "childTreeNode": [{
                    "data": {
                        "menuName": "项目",
                        "menuCode": "BusProject",
                    },
                    "childTreeNode": []
                }, {
                    "data": {
                        "menuName": "我的任务",
                        "menuCode": "BusProject",                                                                              },
                    "childTreeNode": []
                }, {
                    "data": {
                        "menuName": "人员周报",
                        "menuCode": "BusProject",
                    },
                    "childTreeNode": []
                }]
            }, {
                "id": "2",
                "data": {
                    "menuName": "数据统计",
                    "menuCode": "BusClock",
                },
                "childTreeNode": []
            }, {
                "id": "3",
                "data": {
                    "menuName": "人事管理",
                    "menuCode": "",
                },
                "childTreeNode": []
            }, {
                "id": "4",
                "data": {
                    "menuName": "基础管理",
                    "menuCode": "",
                },
                "childTreeNode": []
            }]

朋友们上面看到了,数据data是一个数组,数组里面有好几项数据,所以我们第一步想到的是用 v-for这个指令,将数据一个个都给for出来,于是咱们说干就干。
大家对于上面的这个代码应该多少能看懂,我觉得唯一有疑问的应该是 为啥v-for 仍在了 ul 这个标签上了呢。这里我这么解释。我们的数组中(拿一级的举例你就懂了),每个数据都是一项,然而这每一项我们不知道它到底是有子分类的还是没有子分类的,这个不是我们说了算,而是通过接口获取到的数据说了算。那么,这一项到底有没有子分类就无法判断。这样的话,我们的每一项应该都是一个ul。

我们这么分析,假设我们的分类没有子项,那么属于它的那个ul中的第一个li只要显示它对应的数据就好啦,就像下面

    {{model.data.menuName}}

这是不是很简单呀

情况二,假设我们循环到的那一项有子类,也就是对象中的childTreeNode有数据,那么这个时候,我们是不是就应该把这些数据 '递归出来' 了呢,并且递归在属于它的那一级菜单下面。重点来了,这可怎么把它递归出来呢?

思路:我们应该有第二个列表项,专门用于判断它的childNodes中有没有对应的数据,假设没有,啥也不做,这点毋庸置疑。假设有,那么想办法,自己调用自己,把childNodes属性中的数据重新传入我们的组件当中,让它根据新的数据遍历出来!!!

文字太潦草,我画了张图

子类属性child变成了childTreeNode,data变成dataList什么的不要去纠结,都是一样的东西,只是变名称而已,请重点看我的红线以及我标注的文字

image

在图中你们可以看到,有个tree组件,他就是我们的递归组件了。你随便瞟一眼,有没有发现很简单,不就是我上面说的那个东西吗。然而你可能发现了,我在父组件teamoutput中引用了它,并且将数据传到了他的model属性。

*****在显示完对应数据之后,下面又有个ul标签,并且这个ul标签,把我们对应的childTreeNode属性中的数据,给传进去了。更加重要的是,这个组件items就是组件tree本身,也就是说,我们在这里自己调用了自己,也就是递归了。我们是随便去递归的么?当然不是。我们在决定递归之前,先去用计算属性去判断了childTreeNode属性是否有子类数据,当它有的时候,我们踩决定将数据传入,实现递归,不然的画,它的v-if是false,那么就不会显示。

讲到这里,我相信各位心里都有点BS了吧。

希望我的讲解对你有帮助。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,547评论 6 477
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,399评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,428评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,599评论 1 274
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,612评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,577评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,941评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,603评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,852评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,605评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,693评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,375评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,955评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,936评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,172评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 43,970评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,414评论 2 342

推荐阅读更多精彩内容

  • Android 自定义View的各种姿势1 Activity的显示之ViewRootImpl详解 Activity...
    passiontim阅读 171,398评论 25 707
  • Spring Cloud为开发人员提供了快速构建分布式系统中一些常见模式的工具(例如配置管理,服务发现,断路器,智...
    卡卡罗2017阅读 134,585评论 18 139
  • 我躺在我寝室的床上,听外面午夜十二点以后依然闷热的夏季的风的声音,体会的中国北方在这个时间段热到离谱的接近四...
    50ml阅读 132评论 2 1
  • 第一眼看到这个封面的时候,我的内心是充满了拒绝的…毕竟它的封面,简直就是热门成功学书籍的标配啊!所以估计就算每天去...
    RAINSTOP的游乐场阅读 444评论 0 0
  • 想找一个谈谈心里话的人,没有,因为我没有前度。
    幸运的蝌蚪阅读 569评论 0 1