sql 函式實現三種父子遞迴
在實際運用中經常會建立這樣的結構表Category(Id, ParentId, Name),特別是用於樹形結構時(選單樹,許可權樹..),這種表設計自然而然地會用到遞迴,若是在程式中進行遞迴(雖然在程式中遞迴真的更方便一些),無論是通過ADO.NET簡單sql查詢還是ORM屬性關聯都會執行多次sql語句,難免會造成一些效能上的損耗,所以乾脆使用sql的函式來解決這個問題,用函式返回我們最終需要的結果。
針對這類需求,這裡我列出三種常用的遞迴:
- 以一個節點為基點,列出所有子節點直到無子 (找下級) 。這有點兒像點兵點將,主帥只有一個,下面是左將、右將,左將下面又有千夫長、百夫長,點兵時主帥下令集合,下面的將軍只管各自的隊伍。
- 以一個節點為基點,列出所有父節點直到祖先(找上級) 。
- 麵包屑導航資料(單條資料)
下面我以一幅圖列出這三種形式(實線表現的是我們最終想要的資料,第三幅圖中只有一條資料):
OK,現在讓我們來實現這幾個需求,step by step。
1. 資料準備
根據上面的圖中的資料建立表結構和測試資料
create table Region ( Id int identity primary key, Name nvarchar(20), ParentId int ) go insert into Region(Name,ParentId) values('廣東',NULL) insert into Region(Name,ParentId) values('深圳',1) insert into Region(Name,ParentId) values('惠州',1) insert into Region(Name,ParentId) values('羅湖區',2) insert into Region(Name,ParentId) values('福田區',2) insert into Region(Name,ParentId) values('龍崗區',2) insert into Region(Name,ParentId) values('惠陽區',3) insert into Region(Name,ParentId) values('龍門縣',3) insert into Region(Name,ParentId) values('華強北',5) insert into Region(Name,ParentId) values('體育館',5) select * from Region
2. 正向遞迴實現
/* * summary:遞迴獲取所有子節點 */ alter function GetRecursiveChildren ( @Id int ) returns @t table(Id int,ParentId int,Name nvarchar(20), [Level] int) begin declare @i int set @i = 1 --根節點,Level = 0 insert into @t select @Id,@id,(select Name from Region where Id = @id),0 --直屬子節點,Level = 1 insert into @t select Id,ParentId,Name,@i from Region where ParentId = @Id --如果沒有新的值插入,迴圈結束 while @@rowcount<>0 begin set @i = @i + 1; insert into @t select a.Id,a.ParentId,a.Name,@i from Region a, @t b where a.ParentId = b.Id and b.Level = @i - 1 end return end go --呼叫函式 select * from GetRecursiveChildren(2)
執行上面的函式得到如下圖的結果:
-----------------------------------------------------------------------------------------------------------------------------
當然自sql 2005後微軟提供了CTE(公用表表示式)也可以用於遞迴查詢,請參閱使用公用表示式的遞迴查詢。
上面的遞迴用CTE的sql程式碼如下:
declare @id int set @id = 2 ;with t as--如果CTE前面有語句,需要用分號隔斷 ( select Id, ParentId, Name from Region where Id = @id union all select r1.Id,r1.ParentId,r1.Name from Region r1 join t as r2 on r1.ParentId = r2.Id ) select * from t order by Id
3. 逆向遞迴實現
create function GetRecursiveParent ( @Id int ) returns @t table(Id int,ParentId int,Name nvarchar(20), [Level] int) as begin declare @i int set @i = 1 --插入末節點,Level = 0 insert into @t select @Id,@id,(select Name from Region where Id = @id),0 --插入末節點的父節點,Level = 1 insert into @t select Id,ParentId,Name,@i from Region where Id = (select ParentId from Region where Id = @Id) --如果沒有新的值插入,迴圈結束 while @@rowcount<>0 begin set @i = @i + 1; insert into @t select a.Id,a.ParentId,a.Name,@i from Region a, @t b where a.Id = b.ParentId and b.Level = @i - 1 end return end go --呼叫函式 select * from GetRecursiveParent(10) go
執行這個函式得到的結果如下:
4. 麵包屑實現
create function GetLevel ( @Id int ) returns @level table(IdLevel varchar(100),NameLevel nvarchar(200)) as begin declare @IdLevel varchar(100),@NameLevel nvarchar(200),@Name nvarchar(50) select @IdLevel = cast(@Id as varchar(10)) select @NameLevel = (select Name from Region where Id = @Id) while(exists(select Id,ParentId from Region where Id = (select ParentId from Region where Id = @Id))) begin select @Id = Id,@Name = Name from Region where Id = (select ParentId from Region where Id = @Id) select @IdLevel = cast(@Id as varchar(10)) + '>' + @IdLevel select @NameLevel = @Name + '>' + @NameLevel end insert into @level select @IdLevel,@NameLevel return end go --呼叫函式 select * from GetLevel(10) go
呼叫這個函式的結果如下:
本文sql原始碼下載:http://files.cnblogs.com/keepfool/region_recursive.zip
相關文章
- php 遞迴函式的三種實現方式PHP遞迴函式
- 遞迴函式遞迴函式
- 兩種遞迴方式實現迴文字遞迴
- 三種方法實現strlen函式函式
- 函式表示式–遞迴函式遞迴
- 函式的遞迴函式遞迴
- JavaScript 函式遞迴JavaScript函式遞迴
- php遞迴函式PHP遞迴函式
- 函式之遞迴函式遞迴
- 遞迴匿名函式手動實現 http_build_query 系統函式遞迴函式HTTPUI
- SQL with as 的用法 以及遞迴函式的寫法 遞迴層次查詢SQL遞迴函式
- php利用遞迴函式實現無限級分類PHP遞迴函式
- 函式遞迴與生成式函式遞迴
- 遞迴函式的理解遞迴函式
- 遞迴、三元表示式、生成式(列表,字典)、匿名函式遞迴函式
- 第 8 節:函式-匿名函式、遞迴函式函式遞迴
- oracle中的遞迴sql查詢 connect by prior實現遞迴Oracle遞迴SQL
- Python 函式進階-遞迴函式Python函式遞迴
- JS函式表示式——函式遞迴、閉包JS函式遞迴
- 遞迴函式、演算法之二分法、三元表示式、各種生成式、匿名函式遞迴函式演算法
- 1.5.6 python遞迴函式Python遞迴函式
- 13.0、python遞迴函式Python遞迴函式
- day 17 – 1 遞迴函式遞迴函式
- 遞迴函式例項大全遞迴函式
- 遞迴函式-樹形列表遞迴函式
- go 陣列傳遞給函式三種方式Go陣列函式
- Sql Server 使用CTE實現遞迴查詢SQLServer遞迴
- 初學 PHP 函式的遞迴PHP函式遞迴
- C#語言函式遞迴C#函式遞迴
- GO語言————6.6 遞迴函式Go遞迴函式
- 直觀理解(尾)遞迴函式遞迴函式
- 斐波那契數列三種實現函式函式
- 棧實現遞迴遞迴
- lambda實現遞迴遞迴
- Sql字串分組Split函式的兩種實現方法SQL字串函式
- python實現二叉樹及其七種遍歷方式(遞迴+非遞迴)Python二叉樹遞迴
- 遞迴函式,可變引數列表遞迴函式
- JavaScript 中匿名函式的遞迴呼叫JavaScript函式遞迴