博客
关于我
强烈建议你试试无所不能的chatGPT,快点击我
jsp实现树状结构
阅读量:6388 次
发布时间:2019-06-23

本文共 2956 字,大约阅读时间需要 9 分钟。

首先,介绍一下树状结构在DB中的存储。

使用二维表,如下图,存储树状结构:

现在,我们的目标是想要把这一树状结构表示成:

由上图可以看出它们之间含有一种层级关系,查看源代码,如下:

现在,算法的思路是,先将树状结构按照list的顺序排列出来,这个顺序其实就是去掉了UL和LI标签的顺序,如:

要实现这个顺序其实很简单。

再来看看我们的树

abcdef其实就是这颗树的先序遍历的结果。

那么,如果现在我们只有一张二维表,那么我们要怎样生成这个先序遍历的结果呢?(不使用递归)

观察这张二维表:

node到fatherNode是一对一映射,而fatherNode到node之间则是一对多映射。

也就是说我们可能不太容易知道某个节点的兄弟节点(需要先确定父节点,再通过父节点query兄弟节点),但是确定某个节点的子节点是非常容易的。

如,节点a的所有子节点,只要query fatherNode=a即可。

因此,我们可以使用广搜的思想来解决这个问题。

广搜就是先遍历离根最近的第一层节点,将这些节点放入queue中,然后从queue头不断取出节点,再遍历这个节点的所有子节点,并将其子节点放入queue的队尾,依次下去,知道所有节点都遍历到为止。

拿我们的树来说,就是

但是,如果直接用广搜这个算法的话,我们得到的序列是:

abdcef

与我们所期待的不同。

所以,改进一下该算法。

为了存储我们的结构,新建一个列表。现在,我们就有两个列表了,一个是广搜用的queue,另一个是存储结果的list。

当广搜每摘掉一个头元素,并将该元素的子节点放入queue中时,就在存储的list中的相应位置插入这些子节点们。那么相应位置是哪里呢?就是被摘掉的这个头元素。

例如,现在queue的形态为 “a” ,那么现在要做的操作是把queue的头,a元素摘掉,并且将其子节点"b d"加入queue中,与此同时,在list中查找到a的位置,并将“b d”添加到其后面。此时,queue的形态变为"b d",而list则变为"a b d"。重复此步骤,现在摘掉queue中的头元素,即b,并且将其子节点(c)放入queue尾部,同时在list的b元素后面添加b的子节点(c),此时,queue的形态变为“d c”,而list的形态变为“a b c d”。依照此方式一直进行到广搜结束,即进行到queue为空。

该算法结束时,就可以得到 a b c d e f  这一我们希望见到的序列了。

但是,这一序列丢失了一项很重要的信息,就是层次信息。

所以,再对算法进行一下改进,我们引入一个结构体

<node, level>

使用 level 属性来记录某个节点所处的层次。

那么level这个属性在什么时候进行赋值呢?

在将某个元素的子元素插入list中时。

如何赋值呢?

这个问题很简单,因为是为某个元素插入子元素,因此,被插入的这些元素的level相当于其父节点的 level + 1

所以,list最后就变成了这样一个结构:

node list的顺序 “abcdef”,以及其level属性都是必须的。

现在,我们既知道了节点的顺序,也知道了节点中各个点的关系,因此就可以构建又ul和li表示的树状结构了。

我们再观察一下源代码:

发现,

(1)当level增大时,就会出现一个<ul>标签。

(2)当level减小时,就会出现</ul>标签,而</ul>标签的数量与level减小的程度有关,即减小的level是N,就应该出现N个</ul>标签。

(3)对于每一个list中的item,都由一个<li></li>标签对包围。

由上述规则,我们就可以写出如下程序:

Stack
stack = new Stack
(); String item = ""; String res = ""; ArrayList
list = this.sightTree.getList(); int lastLevel = 0; int currLevel = 0; if(list.size() != 0){ Iterator
iter = list.iterator(); while(iter.hasNext()){ SightLevelPair pair = iter.next(); currLevel = pair.getLevel(); if(currLevel > lastLevel){ res += "
    "; stack.push(item); }else if(currLevel < lastLevel){ int delta = lastLevel - currLevel; while(delta > 0){ res += stack.pop(); --delta; } } res += "
  • " + pair.getSight().getSightName() + "
  • "; lastLevel = currLevel; } while(0 != stack.size()){ res += stack.pop(); } }

可以看到程序中有一个stack,它是用来存储“</ul>”字串的。每当出现一个level增加时,就在最终在字串后加入一个“<ul>”,同时,将一个“</ul>”压入stack中,并且在遇到level减少时,将相应数量的"</ul>"抛出。

结果还是很理想的:

生成这个树用了自定义标签,即定义了一个SimpleTagSupport的子类,用来处理自定义标签。

制作自定义标签的方法。

 

 

转载地址:http://vnbha.baihongyu.com/

你可能感兴趣的文章
Uva 10328 逆向思维
查看>>
(转载)表服务器无法打开与报表服务器数据库的连接。所有请求和处理都要求与数据库建立连接。...
查看>>
团队成员贡献总结
查看>>
MAC下调试JSON接口的工具(HTTP抓包工具)
查看>>
Android设置Gridview中的内容不滚动,然后控件中的内容随便添加的效果。
查看>>
数论 --- 费马小定理 + 快速幂 HDU 4704 Sum
查看>>
(实用)Linux下安装JDK和Eclipse
查看>>
16 包
查看>>
[转]map hash_map unordered_map 性能测试
查看>>
如果点击项目生成时,报错如:不能编辑或者访问拒绝。
查看>>
Selenium学习(一)---Selenium IDE安装及简单介绍
查看>>
PHP控制反转(IOC)和依赖注入(DI)
查看>>
学习计划
查看>>
获取鼠标和元素的坐标点
查看>>
PXE 部署不同版本的系统安装环境以及挽救环境
查看>>
Linux 计划任务
查看>>
flask的orm操作
查看>>
如何防止驱动被恶意利用
查看>>
Nagios的搭建
查看>>
我的友情链接
查看>>