- 1、本文档共11页,可阅读全部内容。
- 2、原创力文档(book118)网站文档一经付费(服务费),不意味着购买了该文档的版权,仅供个人/单位学习、研究之用,不得用于商业用途,未经授权,严禁复制、发行、汇编、翻译或者网络传播等,侵权必究。
- 3、本站所有内容均由合作方或网友上传,本站不对文档的完整性、权威性及其观点立场正确性做任何保证或承诺!文档内容仅供研究参考,付费前请自行鉴别。如您付费,意味着您自己接受本站规则且自行承担风险,本站不退款、不进行额外附加服务;查看《如何避免下载的几个坑》。如果您已付费下载过本站文档,您可以点击 这里二次下载。
- 4、如文档侵犯商业秘密、侵犯著作权、侵犯人身权等,请点击“版权申诉”(推荐),也可以打举报电话:400-050-0827(电话支持时间:9:00-18:30)。
- 5、该文档为VIP文档,如果想要下载,成为VIP会员后,下载免费。
- 6、成为VIP后,下载本文档将扣除1次下载权益。下载后,不支持退款、换文档。如有疑问请联系我们。
- 7、成为VIP后,您将拥有八大权益,权益包括:VIP文档下载权益、阅读免打扰、文档格式转换、高级专利检索、专属身份标志、高级客服、多端互通、版权登记。
- 8、VIP文档为合作方或网友上传,每下载1次, 网站将根据用户上传文档的质量评分、类型等,对文档贡献者给予高额补贴、流量扶持。如果你也想贡献VIP文档。上传文档
查看更多
采用左右值编码来存储无限分级树形结构的数据库表设计之前我介绍过一种按位数编码保存树形结构数据的表设计方法,详情见:浅谈数据库设计技巧(上)
该设计方案的优点是:只用一条查询语句即可得到某个根节点及其所有子孙节点的先序遍历。由于消除了递归,在数据记录量较大时,可以大大提高列表效率。但是,这种编码方案由于层信息位数的限制,限制了每层能所允许的最大子节点数量及最大层数。同时,在添加新节点的时候必须先计算新节点的位置是否超过最大限制。
上面的设计方案必须预先设定类别树的最大层数以及最大子节点数,不是无限分级,在某些场合并不能采用,那么还有更完美的解决方案吗?通过 google的搜索,我又探索到一种全新的无递归查询,无限分级的编码方案——左右值。原文的程序代码是用php写的,但是通过仔细阅读其数据库表设计说明及相关的sql语句,我彻底弄懂了这种巧妙的设计思路,并在这种设计中新增了删除节点,同层平移的需求(原文只提供了列表及插入子节点的sql语句)。
下面我力图用比较简短的文字,少量图表,及相关核心sql语句来描述这种设计方案:
首先,我们弄一棵树作为例子:
商品|---食品|????|---肉类|????|????|--猪肉 |????|---蔬菜类 |??????????|--白菜 |---电器 ?????|--电视机 ?????|--电冰箱
?
采用左右值编码的保存该树的数据记录如下(设表名为tree):
Type_id Name Lft Rgt 1 商品 1 18 2 食品 2 11 3 肉类 3 6 4 猪肉 4 5 5 蔬菜类 7 10 6 白菜 8 9 7 电器 12 17 8 电视机 13 14 9 电冰箱 15 16 ?
第一次看见上面的数据记录,相信大部分人都不清楚左值(Lft)和右值(Rgt)是根据什么规则计算出来的,而且,这种表设计似乎没有保存父节点的信息。下面把左右值和树结合起来,请看:
?1商品18
+---------------------------------------+
??????????????? 2食品11??????????????????????????????????? 12电器17????????? +-----------------+???????????????????? +---------------------+????3肉类6????????? 7蔬菜类10????????? 13电视机14?????? 15电冰箱16??? 4猪肉5?????????? 8白菜9
请用手指指着上图中的数字,从1数到18,学习过数据结构的朋友肯定会发现什么吧?对,你手指移动的顺序就是对这棵树的进行先序遍历的顺序。接下来,让我讲述一下如何利用节点的左右值,得到该节点的父节点,子孙节点数量,及自己在树中的层数。
?
假定我们要对节点“食品”及其子孙节点进行先序遍历的列表,只需使用如下一条sql语句:
select * from tree where Lft between 2 and 11 order by Lft asc
查询结果如下:
Type_id Name Lft Rgt 2 食品 2 11 3 肉类 3 6 4 猪肉 4 5 5 蔬菜类 7 10 6 白菜 8 9 ?
那么某个节点到底有多少子孙节点呢?很简单,子孙总数 =(右值-左值-1)/2?
以节点“食品”举例,其子孙总数=(11-2-1)/ 2 = 4
?
同时,我们在列表显示整个类别树的时候,为了方便用户直观的看到树的层次,一般会根据节点所处的层数来进行相应的缩进,那么,如何计算节点在树中的层数呢?还是只需通过左右值的查询即可,以节点“食品”举例,sql语句如下:
select count(*) from tree where lft = 2 and rgt = 11
为了方便列表,我们可以为tree表建立一个视图,添加一个层数列,该类别的层数可以写一个自定义函数来计算。该函数如下:
CREATE?FUNCTION?dbo.CountLayer(????????@type_id?int)RETURNS?intASbegin????declare?@result?int????set?@result=0????declare?@lft?int????declare?@rgt?int????if?exists?(select?1?from?tree?where?type_id=@type_id)????begin????????select?@lft=lft,@rgt=rgt?from?tree?where?type_
您可能关注的文档
- (17一2)市区教研室推进校本教研工作的经验与成效(校本新讲座一之2).doc
- (17一4)教员在校本教研中的专业引领作用(校本新讲座一之4).doc
- (2012.0)年产1000万台手机等电子产品项目.doc
- (安吉龚海乐)我共参与 家园更美丽.doc
- (表6)油气回治理改造工程施工验收资料记录表.doc
- (附件5)贵州教育科学规划课题申请评审书.doc
- (该申请书为模.doc
- (教师用书)213-2014学年高中物理 第七章第6节 探究功与物体速度变化的关系素材 新人教版必修2.doc
- (结项报告)化学院+王纪学+石墨烯基纳米薄膜的制备及其光电性能研究.doc
- (考试必备)高地理大纲版总复习知识点精析精练与高考题预测:第二部分人文地理 第五单元 第十五讲 农业生.doc
文档评论(0)