(转)父子节点遍历

原创作品,允许转载,转载时请务必以超链接形式标明文章 原始出处 、作者信息和本声明。否则将追究法律责任。http://qianzhang.blog.51cto.com/317608/1205679

对于树/图的遍历,通常有2种算法来实现:迭代(Iteration)和递归(Recursion),迭代就是利用循环反复取值/赋值的过程;递归则是反复自己调用自己来获得最终结果。

不过SQL Server里的递归有32层嵌套限制,目的在于防止代码进入死循环。除非使用提示OPTION (MAXRECURSION 0)

测试数据:

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
if OBJECT_ID('city'is not null
    drop table city
GO
create table city
(
id    int,
name  nvarchar(10),
pid   int,
depth int
)
GO
insert into city
select  1,N'江苏省',0,0 union all
select  2,N'南京市',1,1 union all
select  3,N'玄武区',2,2 union all
select  4,N'鼓楼区',2,2 union all
select  5,N'浙江省',0,0 union all
select  6,N'杭州市',5,1 union all
select  7,N'西湖区',6,2 union all
select  8,N'滨江区',6,2 union all
select  9,N'苏州市',1,1 union all
select 10,N'吴中区',9,2 union all
select 11,N'吴江区',9,2

 

一. 查找子节点

查找节点1的所有子节点,返回结果如下:

idnamepiddepth
1江苏省00
2南京市11
3玄武区22
4鼓楼区22
9苏州市11
10吴中区92
11吴江区92

 

1. 迭代

(1) 不借助depth,通过not in来向下查找

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
if OBJECT_ID('f_get_child'is not null
    drop function f_get_child
GO
create function f_get_child
(
@id int
)
returns @t table(id int)
as
begin
    insert into @t select @id
    --insert into @t select id from city where pid = @id
    while @@ROWCOUNT>0
    begin
        insert into @t
        select a.id
        from city a inner join @t b on a.pid = b.id
        where a.id not in(select id from @t)
    end
    return
end
GO
select from city where id in(select id from f_get_child(1))

 

(2) 通过depth来逐层查找

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
if OBJECT_ID('f_get_child'is not null
    drop function f_get_child
GO
create function f_get_child
(
@id int
)
returns @t table(id int, depth int)
begin
    declare @depth int
    set @depth = 0
    insert @t select ID,@depth from city where ID =@ID
    while @@ROWCOUNT>0
    begin
        set @depth = @depth + 1
        insert @t select a.ID,@depth
          from city a, @t b
         where a.pid = b.ID
           and b.depth = @depth - 1
    end
    return 
end
GO
select from city where id in(select id from f_get_child(1))

 

2. 递归

(1) 自定义函数递归

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
if OBJECT_ID('f_get_child'is not null
    drop function f_get_child
GO
create function f_get_child
(
@id int
)
returns @t table(id int)
as
begin
    declare @pid int
    set @pid = null
    insert into @t
    select @id
    union all
    select id from city where pid = @id
                                                                      
    if exists(select 1
        from city a inner join @t b on a.pid = b.id
        where a.id not in(select id from @t))
    begin
        insert into @t
        select a.id
        from city a inner join @t b on a.pid = b.id
        where a.id not in(select id from @t)
        union all
        select from f_get_child(@pid)
    end
    return
end
GO
select from city where id in(select from f_get_child(1))

 

(2) CTE递归

1
2
3
4
5
6
7
8
9
10
11
12
declare @id int
set @id = 1;
with tmp
as
(
select from city where id = @id
union all
select a.* from city a
inner join tmp b
on a.pid = b.id
)
select from tmp order by id

 

二. 查找父节点

查找节点8的所有父节点,返回结果如下:

idnamepiddepth
5浙江省00
6杭州市51
8滨江区62

 

1. 迭代

父节点只有一个,不需要做什么限制,一直往上级查找pid就可以了

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
if OBJECT_ID('f_get_parent'is not null
    drop function f_get_parent
GO
create function f_get_parent
(
@id int
)
returns @t table(id int)
as
begin
    declare @pid int
    insert into @t select @id
    select @pid = pid from city where id = @id
    while @pid<>0
    begin
        insert into @t values(@pid)
        select @pid=pid from city where id=@pid
    end
    return
end
GO
select from city where id in(select from f_get_parent(8))

 

2. 递归

(1) 自定义函数递归

1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
if OBJECT_ID('f_get_parent'is not null
    drop function f_get_parent
GO
create function f_get_parent(@id int)
returns @t table(id int)
AS
begin
    declare @pid int
    select top 1 @pid = pid
    from city
    where id = @id
    if @pid <> 0
    begin
        insert into @t
        select @id
        union all
        select from f_get_parent(@pid)
    end
    else
    begin
        insert into @t
        select @id
    end
    return
end
GO
select from city where id in(select from f_get_parent(8))

 

(2) CTE递归

1
2
3
4
5
6
7
8
9
10
11
12
declare @id int
set @id = 8;
with tmp
as
(
select from city where id = @id
union all
select a.* from city a
inner join tmp b
on a.id = b.pid
)
select from tmp order by id

转载于:https://www.cnblogs.com/Luouy/p/3173989.html

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/261445.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

SharePoint 2007 做Migration后用户处理

新公司的一部分任务是做Migration&#xff0c;从用户角度来说分为两种情况&#xff0c;第一种是在同一个Domain下作网站的Migration&#xff0c;另外一种就是在不同Domain下做Migration。在同一Domain下做Migration&#xff0c;User不会出现问题&#xff0c;因为在Import后&…

Ubuntu GNOME 15.10升级16.4LTS

为什么80%的码农都做不了架构师&#xff1f;>>> 当Ubuntu GNOME官方已经发送16.4后&#xff0c;执行以下命令 sudo do-release-upgrade 显示没有新系统更新 在网上找到另一种方法是执行以下命令&#xff1a; sudo ppa-purge ppa:gnome3-team/gnome3-staging sudo p…

matplotlib plot 分组_小白学 Python 数据分析(16):Matplotlib(一)坐标系

人生苦短&#xff0c;我用 Python前文传送门&#xff1a;小白学 Python 数据分析(1)&#xff1a;数据分析基础小白学 Python 数据分析(2)&#xff1a;Pandas (一)概述小白学 Python 数据分析(3)&#xff1a;Pandas (二)数据结构 Series小白学 Python 数据分析(4)&#xff1a;Pa…

hdu 1423 最长公共递增子序列

这题一开始把我给坑了&#xff0c;我还没知道LCIS的算法&#xff0c;然后就慢慢搞吧&#xff0c;幸运的是还真写出来了&#xff0c;只不过麻烦了一点。 我是将该题转换为多条线段相交&#xff0c;然后找出最多多少条不相交&#xff0c;并且其数值死递增的。 代码如下&#xff1…

安卓从业者应该关注:Android 6.0的运行时权限

Android 6.0&#xff0c;代号棉花糖&#xff0c;自发布伊始&#xff0c;其主要的特征运行时权限就很受关注。因为这一特征不仅改善了用户对于应用的使用体验&#xff0c;还使得应用开发者在实践开发中需要做出改变。 没有深入了解运行时权限的开发者通常会有很多疑问&#xff0…

git把另一个盘的代码上传_如何使用Git上传代码到GitHub

初始化Git1. 安装Git客户端一路下一步下一步既可2. 配置Git1. 在电脑硬盘里找一块地方存放本地仓库,右键Git Bash进入git命令行执行如下代码 生成.git文件夹,表示本地git创建成功git config --global user.name "littonishir"git config --global user.email "l…

sh脚本学习之: sh脚本 、sed、awk

sh脚本 sh命令的批处理文件&#xff0c;支持更复杂的逻辑。  Shell中的变量 参数   $0  当前脚本路径 $1....$n  脚本执行对应的第n个参数 条件判断   文件判断 test [op] path e存在  f是文件  d是目录  r可读  w可写  x可执行   if判断/case if[条件判…

Silverlight OOB Setup

Sl的安装大家都知道&#xff1a;在程序点右键&#xff0c;选择安装。有没有另外一种方法&#xff0c;类似普通软件那样&#xff0c;通过安装包安装&#xff1f;答案当然是 有参考&#xff1a;http://www.code Technorati 标签: project.com/KB/install/slsetup.aspx思路很简单&…

2016-04-29 二分查找的面试题

为什么80%的码农都做不了架构师&#xff1f;>>> 1.面试题 例如&#xff1a; ip计算后的值&#xff1d;53文本内容&#xff1a;1,100,北京 101,1000,上海 1001,3001,广州 ...求ip53对应的省份2.代码如下&#xff1a; #!/usr/bin/python # coding: utf8def ip_find(i…

gettype拿不到值_王者荣耀:被低估的强势打野,就是这位拿大锤子的阿姨!

王者峡谷的小伙伴你们好&#xff0c;今天就为你们推荐一下这位野区女霸主钟无艳&#xff0c;不仅高伤害而且操作简单&#xff01;版本更新在5月14日版本更新中&#xff0c;钟无艳的三个技能都被加强了&#xff0c;所有的蓝耗都被固定&#xff0c;不再随技能等级的成长值&#x…

如何培训

Ground roles 1>No phone,no computer.2>没有私下讨论 3>parkng Lot 1. 演讲技巧 2. 隐喻 说服别人时&#xff0c;讲故事&#xff0c;举例子 3. 问题回答 4. 会场设置 转载于:https://www.cnblogs.com/myfav/p/3176621.html

【笔记】MATLAB中的图形(2)

三维作图 1、mesh(z)语句 mesh(z)语句可以给出矩阵z元素的三维消隐图&#xff0c;网络表面由z坐标点定义&#xff0c;与前面叙述的x-y平面的线格相同&#xff0c;图形由临近的点连接而成。它可用来显示用其他方式难以输出的包含大量数据的大型矩阵&#xff0c;也可以用来绘制z变…

Kindeditor放置两个调用readonly错误

开始 需要调用Kindeditor中的readonly的方法&#xff0c;但是一直提示edit is undefined 而editor.readonly(true)又只对第一个对象有效 所以只能换换形式&#xff0c;干脆将下面的kindeditor拿上来 虽然是满足自己这个需求&#xff0c;但是真正的原因解决办法&#xff0c;还是…

acl在内核里的位置_Linux 进程在内核眼中是什么样子的?

本篇算是进程管理的的揭幕篇&#xff0c;简单介绍一个进程在内核眼里的来龙去脉&#xff0c;为接下来的进程创建&#xff0c;进程调度&#xff0c;进程管理等篇章做好学习准备。从程序到进程再到内核啥是程序&#xff0c;啥是进程&#xff0c;一张图可以给我们解释&#xff1a;…

majikan

转载于:https://www.cnblogs.com/YOUEN/p/3179091.html

sql语句查询数据库返回结果转换显示自定义字段

在开发中经常遇到在数据中用单字符保存数据对应简单信息&#xff0c;比如性别、状态、与否等。如果要求在绑定数据源并显示对应字段&#xff0c;比如性别&#xff1a;1表示男&#xff0c;0表示女&#xff1b;状态&#xff1a;1表示有效&#xff0c;0表示失效等等。简单一句sql语…

经典中的品味:第一章 C++的Hello,World!

“程序设计要通过编写程序的实践来学习”—Brian Kernighan 1.1 程序 何为程序?简单的说&#xff0c;就是为了使计算机能够做事&#xff0c;你需要在繁琐的细节中告诉它怎么做。对于怎么做的描述就是程序。编程是书写和测试怎么做的过程。维基百科上说&#xff0c;一个程序就像…

本地数据jqGrid分页

var mydata; $(function() { var str ; str "<span>共<span idp_total></span>条记录</span> " str "<span>每页" "<select idset_page_size classui-button ui-widget ui-state-default ui-corner-all&g…

避免将项目名称用作映射类型名称

避免将项目名称用作映射类型名称 在 Visual Studio 中向 BizTalk 项目添加新映射时&#xff0c;请不要将项目名称用作类型名称。如果这样做&#xff0c;编译器将生成一个或多个错误&#xff0c;类似于“类型中不存在类型名称‘’”。 若要从 BizTalk 项目内更改映射的类型名称&…

大教堂和市集

该文的作者Eric Raymond是Open Source Software领域的领袖&#xff0c;这方面许多新的思想正是从他那儿产生的&#xff0c;同时他也是UNIX上最流行的Email软件Fetchmail 的作者。 HansB翻译 本博文仅做简单排版 一. 大教堂和市集 Linux的影响是非常巨大的。甚至在&#xff15…