树形结构数据存储方案(三):闭包表

2017 年 9 月 1 日 数据库开发

(点击上方公众号,可快速关注)


来源:标点符

www.biaodianfu.com/closure-table.html

如有好文章投稿,请点击 → 这里了解详情


将Closure Table翻译成闭包表不知道是否合适,闭包表的思路和物化路径差不多,都是空间换时间,Closure Table,一种更为彻底的全路径结构,分别记录路径上相关结点的全展开形式。能明晰任意两结点关系而无须多余查询,级联删除和结点移动也很方便。但是它的存储开销会大一些,除了表示结点的Meta信息,还需要一张专用的关系表。


以下图举例数据举例:



创建主表:


CREATE TABLE nodeInfo (

node_id INT NOT NULL AUTO_INCREMENT,

node_name VARCHAR (255),

PRIMARY KEY (`node_id`)

) DEFAULT CHARSET = utf8;


创建关系表:


CREATE TABLE nodeRelationship (

ancestor INT NOT NULL,

descendant INT NOT NULL,

distance INT NOT NULL,

PRIMARY KEY (ancestor, descendant)

) DEFAULT CHARSET = utf8;


其中


  • Ancestor代表祖先节点

  • Descendant代表后代节点

  • Distance 祖先距离后代的距离


添加数据(创建存储过程)


CREATE DEFINER = `root`@`localhost` PROCEDURE `AddNode`(`_parent_name` varchar(255),`_node_name` varchar(255))

BEGIN

DECLARE _ancestor INT;

DECLARE _descendant INT;

DECLARE _parent INT;

IF NOT EXISTS(SELECT node_id From nodeinfo WHERE node_name = _node_name)

THEN

INSERT INTO nodeinfo (node_name) VALUES(_node_name);

SET _descendant = (SELECT node_id FROM nodeinfo WHERE node_name = _node_name);

INSERT INTO noderelationship (ancestor,descendant,distance) VALUES(_descendant,_descendant,0);

IF EXISTS (SELECT node_id FROM nodeinfo WHERE node_name = _parent_name)

THEN

SET _parent = (SELECT node_id FROM nodeinfo WHERE node_name = _parent_name);

INSERT INTO noderelationship (ancestor,descendant,distance) SELECT ancestor,_descendant,distance+1 from noderelationship where descendant = _parent;

END IF;

END IF;

END;


完成后2张表的数据大致是这样的:(注意:每个节点都有一条到其本身的记录。)




查询Fruit下所有的子节点:


SELECT

n3.node_name

FROM

nodeinfo n1

INNER JOIN noderelationship n2 ON n1.node_id = n2.ancestor

INNER JOIN nodeinfo n3 ON n2.descendant = n3.node_id

WHERE

n1.node_name = 'Fruit'

AND n2.distance != 0


查询Fruit下直属子节点:


SELECT

n3.node_name

FROM

nodeinfo n1

INNER JOIN noderelationship n2 ON n1.node_id = n2.ancestor

INNER JOIN nodeinfo n3 ON n2.descendant = n3.node_id

WHERE

n1.node_name = 'Fruit'

AND n2.distance = 1


查询Fruit所处的层级:


SELECT

n2.*, n3.node_name

FROM

nodeinfo n1

INNER JOIN noderelationship n2 ON n1.node_id = n2.descendant

INNER JOIN nodeinfo n3 ON n2.ancestor = n3.node_id

WHERE

n1.node_name = 'Fruit'

ORDER BY

n2.distance DESC


另外要删除节点也非常的简单,这里就不再做过多的阐述。


参考链接:


  • https://coderwall.com/p/lixing/closure-tables-for-browsing-trees-in-sql



看完本文有收获?请转发分享给更多人

关注「数据库开发」,提升 DB 技能

登录查看更多
0

相关内容

[ICML2020]层次间消息传递的分子图学习
专知会员服务
33+阅读 · 2020年6月27日
【北京大学】面向5G的命名数据网络物联网研究综述
专知会员服务
36+阅读 · 2020年4月26日
Transformer文本分类代码
专知会员服务
116+阅读 · 2020年2月3日
【论文】结构GANs,Structured GANs,
专知会员服务
14+阅读 · 2020年1月16日
知识图谱本体结构构建论文合集
专知会员服务
106+阅读 · 2019年10月9日
已删除
AI掘金志
7+阅读 · 2019年7月8日
树形结构为什么不需要归一化?
七月在线实验室
8+阅读 · 2019年4月30日
34个最优秀好用的Python开源框架
专知
9+阅读 · 2019年3月1日
中文NLP用什么?中文自然语言处理的完整机器处理流程
人工智能头条
61+阅读 · 2018年9月5日
干货 :数据分析师的完整流程与知识结构体系
数据分析
8+阅读 · 2018年7月31日
在NLP中深度学习模型何时需要树形结构?
全球人工智能
5+阅读 · 2018年3月29日
Neo4j 和图数据库起步
Linux中国
8+阅读 · 2017年12月20日
python pandas 数据处理
Python技术博文
4+阅读 · 2017年8月30日
Caffe 深度学习框架上手教程
黑龙江大学自然语言处理实验室
14+阅读 · 2016年6月12日
Arxiv
45+阅读 · 2019年12月20日
Arxiv
4+阅读 · 2019年1月14日
VIP会员
相关VIP内容
[ICML2020]层次间消息传递的分子图学习
专知会员服务
33+阅读 · 2020年6月27日
【北京大学】面向5G的命名数据网络物联网研究综述
专知会员服务
36+阅读 · 2020年4月26日
Transformer文本分类代码
专知会员服务
116+阅读 · 2020年2月3日
【论文】结构GANs,Structured GANs,
专知会员服务
14+阅读 · 2020年1月16日
知识图谱本体结构构建论文合集
专知会员服务
106+阅读 · 2019年10月9日
相关资讯
已删除
AI掘金志
7+阅读 · 2019年7月8日
树形结构为什么不需要归一化?
七月在线实验室
8+阅读 · 2019年4月30日
34个最优秀好用的Python开源框架
专知
9+阅读 · 2019年3月1日
中文NLP用什么?中文自然语言处理的完整机器处理流程
人工智能头条
61+阅读 · 2018年9月5日
干货 :数据分析师的完整流程与知识结构体系
数据分析
8+阅读 · 2018年7月31日
在NLP中深度学习模型何时需要树形结构?
全球人工智能
5+阅读 · 2018年3月29日
Neo4j 和图数据库起步
Linux中国
8+阅读 · 2017年12月20日
python pandas 数据处理
Python技术博文
4+阅读 · 2017年8月30日
Caffe 深度学习框架上手教程
黑龙江大学自然语言处理实验室
14+阅读 · 2016年6月12日
Top
微信扫码咨询专知VIP会员