CTE递归有序树
CTE Recursion Ordered Tree
我用以下数据
创建了这个SQL Fiddle
userId userName managerId
====== ======== =========
1 Adam NULL
2 Brett 1
3 Chris 2
4 George 1
5 David 3
6 Elliot 5
7 Fred 5
8 Harry 4
我如何 return 树使得数据按以下顺序 returned:
Adam
Brett
Chris
David
Elliot
Fred
George
Harry
我不担心缩进,当然我不能只按名字排序(以防 Fred 更正为 Alfred)。
这是我目前得到的:
WITH UserCTE AS (
SELECT userId, userName, managerId, 0 AS EmpLevel
FROM Users where managerId is null
UNION ALL
SELECT usr.userId, usr.userName, usr.managerId, mgr.[EmpLevel]+1
FROM Users AS usr
INNER JOIN UserCTE AS mgr
ON usr.managerId = mgr.userId where usr.managerId IS NOT NULL
)
SELECT *
FROM UserCTE AS u
ORDER BY EmpLevel;
您需要获取每个人的完整路径,然后按此排序:
WITH UserCTE AS (
SELECT userId, userName, managerId, 0 AS EmpLevel,
CONVERT(VARCHAR(MAX), '/' + userName) as path
FROM Users
WHERE managerId is null
UNION ALL
SELECT usr.userId, usr.userName, usr.managerId, mgr.[EmpLevel]+1,
CONVERT(VARCHAR(MAX), mgr.path + '/' + usr.userName)
FROM Users usr INNER JOIN
UserCTE mgr
ON usr.managerId = mgr.userId
WHERE usr.managerId IS NOT NULL -- this is unnecessary
)
SELECT *
FROM UserCTE AS u
ORDER BY path;
如何使用 sql server hierarchyid 以正确的顺序对它们进行排序:
SQL Fiddle
MS SQL Server 2014 架构设置:
CREATE TABLE [dbo].[Users](
[userId] [int] ,
[userName] [varchar](50) ,
[managerId] [int] ,
)
INSERT INTO dbo.Users
([userId], [userName], [managerId])
VALUES
(1,'Adam',NULL),
(2,'Brett',1),
(3,'Chris',2),
(4,'George',1),
(5,'David',3),
(6,'Elliot',5),
(7,'Frank',5),
(8,'Harry',4)
查询 1:
WITH UserCTE AS (
SELECT userId, userName, managerId, hierarchyid::GetRoot() AS EmpLevel
FROM Users where managerId is null
UNION ALL
SELECT usr.userId, usr.userName, usr.managerId
, cast(mgr.EmpLevel.ToString() + cast(usr.userId As varchar(30)) + '/' as hierarchyid) as EmpLevel
FROM Users AS usr
INNER JOIN UserCTE AS mgr
ON usr.managerId = mgr.userId where usr.managerId IS NOT NULL
)
SELECT * , EmpLevel.ToString()
FROM UserCTE AS u
ORDER BY EmpLevel
| userId | userName | managerId | EmpLevel | |
|--------|----------|-----------|----------|-----------|
| 1 | Adam | (null) | | / |
| 2 | Brett | 1 | aA== | /2/ |
| 3 | Chris | 2 | a8A= | /2/3/ |
| 5 | David | 3 | a+M= | /2/3/5/ |
| 6 | Elliot | 5 | a+OU | /2/3/5/6/ |
| 7 | Frank | 5 | a+Oc | /2/3/5/7/ |
| 4 | George | 1 | hA== | /4/ |
| 8 | Harry | 4 | hog= | /4/8/ |
我用以下数据
创建了这个SQL FiddleuserId userName managerId
====== ======== =========
1 Adam NULL
2 Brett 1
3 Chris 2
4 George 1
5 David 3
6 Elliot 5
7 Fred 5
8 Harry 4
我如何 return 树使得数据按以下顺序 returned:
Adam
Brett
Chris
David
Elliot
Fred
George
Harry
我不担心缩进,当然我不能只按名字排序(以防 Fred 更正为 Alfred)。
这是我目前得到的:
WITH UserCTE AS (
SELECT userId, userName, managerId, 0 AS EmpLevel
FROM Users where managerId is null
UNION ALL
SELECT usr.userId, usr.userName, usr.managerId, mgr.[EmpLevel]+1
FROM Users AS usr
INNER JOIN UserCTE AS mgr
ON usr.managerId = mgr.userId where usr.managerId IS NOT NULL
)
SELECT *
FROM UserCTE AS u
ORDER BY EmpLevel;
您需要获取每个人的完整路径,然后按此排序:
WITH UserCTE AS (
SELECT userId, userName, managerId, 0 AS EmpLevel,
CONVERT(VARCHAR(MAX), '/' + userName) as path
FROM Users
WHERE managerId is null
UNION ALL
SELECT usr.userId, usr.userName, usr.managerId, mgr.[EmpLevel]+1,
CONVERT(VARCHAR(MAX), mgr.path + '/' + usr.userName)
FROM Users usr INNER JOIN
UserCTE mgr
ON usr.managerId = mgr.userId
WHERE usr.managerId IS NOT NULL -- this is unnecessary
)
SELECT *
FROM UserCTE AS u
ORDER BY path;
如何使用 sql server hierarchyid 以正确的顺序对它们进行排序: SQL Fiddle
MS SQL Server 2014 架构设置:
CREATE TABLE [dbo].[Users](
[userId] [int] ,
[userName] [varchar](50) ,
[managerId] [int] ,
)
INSERT INTO dbo.Users
([userId], [userName], [managerId])
VALUES
(1,'Adam',NULL),
(2,'Brett',1),
(3,'Chris',2),
(4,'George',1),
(5,'David',3),
(6,'Elliot',5),
(7,'Frank',5),
(8,'Harry',4)
查询 1:
WITH UserCTE AS (
SELECT userId, userName, managerId, hierarchyid::GetRoot() AS EmpLevel
FROM Users where managerId is null
UNION ALL
SELECT usr.userId, usr.userName, usr.managerId
, cast(mgr.EmpLevel.ToString() + cast(usr.userId As varchar(30)) + '/' as hierarchyid) as EmpLevel
FROM Users AS usr
INNER JOIN UserCTE AS mgr
ON usr.managerId = mgr.userId where usr.managerId IS NOT NULL
)
SELECT * , EmpLevel.ToString()
FROM UserCTE AS u
ORDER BY EmpLevel
| userId | userName | managerId | EmpLevel | |
|--------|----------|-----------|----------|-----------|
| 1 | Adam | (null) | | / |
| 2 | Brett | 1 | aA== | /2/ |
| 3 | Chris | 2 | a8A= | /2/3/ |
| 5 | David | 3 | a+M= | /2/3/5/ |
| 6 | Elliot | 5 | a+OU | /2/3/5/6/ |
| 7 | Frank | 5 | a+Oc | /2/3/5/7/ |
| 4 | George | 1 | hA== | /4/ |
| 8 | Harry | 4 | hog= | /4/8/ |