树结构(即无限级分类),使用左右值的树模型——优化方案

前言:

“子又生孙,孙又生子;子又有子,子又有孙。子子孙孙,无穷匮也”——《愚公移山》

树是由根而来,繁衍出来的称之为节点,根也是节点。如下图,使用左右序号对每一个节点进行标志。根节点的左序是树的最小序号,根节点的右序是树的最大序号,且高度是0。

常见树模型结构

数据表设计:

看懂上图节点的表示之后,我们来看看数据表的设计。上图的每个节点的号码是id,left_no, right_no不必解释,与上图示图一一对应,high表示该节点的高度,0只有一个节点,那就是根节点。p_id字段表示当前节点的父级id,有人觉得这个p_id有点冗余,其实这个p_id大有用处。

数据表设计

1. 查询操作:

1.1 查询某个节点的后代

在上图的模型,观察任意节点,其后代的左序一定比当前节点左序大,后代的右序一定比当前节点右序小,那么依据这个条件我们就可以查出当前节点的后代,sql语句如下:

查询后代sql语句

在实际开发中,查出来的数据,我们需要对其组成树结构。也就是嵌套成数组或者对象的结构。讲一下表字段的作用, high这个字段可以控制查询树的高度,例如我们只需要三层(不包括当前节点,即四层),sql语句如下:

查询指定层级后代sql语句

查出来的每一条数据都带有p_id字段,这个字段对于组成对象或者数组结构起着至关重要的作用。遍历数据,每一个节点的子对象或数组都要依据父节点的id来获取p_id对应的子节点,至于代码如何具体的实现,这里不做展示,心领神会即可。

1.2 查询某个节点的祖先

在上图的模型,观察任意节点,其祖先的左序一定比当前节点左序小,其祖先的右序一定比当前节点右序大,sql语句如下:

查询祖先sql语句

high字段对于控制查询多少层祖先也有作用,与上面相似,不做赘述。

1.3 查询某个节点的兄弟节点

此处举例两条sql示例:

查询兄弟节点sql语句
查询兄弟节点sql语句

2. 添加、移动操作

2.1 在任意节点添加一个新节点

步骤一(腾位):左序大于当前节点右序的左右序号都自增2,sql如下:

腾位sql语句

步骤二(入位):新节点的左序等于当前节点的右序,新节点的右序等于当前节点的右序+1,sql如下:

入位sql语句

同时,也要把当前节点的序号自增2

自增sql语句

2.2 移动操作

常见树模型

注意:

1. 更换操作相对复杂,为保证更换操作不出错,建议在新增节点的地方设置开关,比如在redis设置某个值,是否允许新增节点,0表示不允许,表示当前有正在更换操作,1表示允许,表示当前没有更换操作在进行,所以在开始更换操作之前先设置0

2. 补充设计表,数据表多加一个字段,表示该数据是否已经删除,status 0-删除,1-正常。


需要将id=6的节点整个团队更换到id=5的节点下,

步骤一: 将id为6的团队从树结构删除,即设置status=0

删除指定团队

步骤二:将左节点大于id=6节点右序自减团队人数*2,(请注意看各节点的左右序变化)

回位一

步骤三:id=6的祖先的右序自减团队人数的两倍,

回位二

步骤四:

    腾位,左节点大于id=5的右序,且status=1,左右序自增新团队人数*2

腾位一

id=5的祖先的右序自增新团队人数*2

腾位二

步骤五:

归位,新团队人员左右序编号,这里的编号有技巧,因为前面删除id=6的团队,只是标志status=0,被删除的团队各个节点的左右序还保留,现在只需要计算新父节点的右序与id=6的节点左序,然后id=6的节点和其团队成员自增或自减差值、将status设置为1。

归位

步骤六:

    新父级节点更新

更新父节点

至此,整个更换的过程就完成了。


可是,每次一更换需要更新大量的节点,所以此处提出优化方案

树结构模型

我们可以先观察一下上图,将id=6团队更换到id=5的节点下,更换后,id=2, id=3, id=2, id=4, id=5, id=6, id=7, id=8, id=10的左右序是没有变化的,因为控制了这个过程不会有新的节点进来,数量是一定的,id=2这个节点是id=5,id=6的公共祖先节点,我们只需要更新该公共祖先节点下面的各个节点左右序即可,当这棵树庞大的时候,这种节制的操作可以大大减少不相关的数据操作了。

楼主在业务上实践这个方案的时候,已经做过建模并且测试,如果你有采用这种方案的话,建议多多建模,考虑多种情况,并且更换节点的时候需要多层判断,比如不允许id=6的节点更换到id=9的节点下。


具体的sql语句不展示,毕竟这篇文章不是教新手如何写sql。

本文章仅是个人的总结,不好的地方还请多多指正。

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念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

推荐阅读更多精彩内容