Допустим, у меня есть следующая таблица:
CREATE TABLE Employees
(
EmployeeId int PRIMARY KEY NOT NULL,
ParentEmployeId int REFERENCES Employees(EmployeeId) NULL,
Name varChar(255)
)
Все записи имеют первичный идентификатор, и записи могут идентифицировать другую запись как родительскую. (Моя фактическая схема не о сотрудниках, это просто упрощенная версия для иллюстрации, поэтому, если у вас есть лучший способ обработки информации о сотрудниках, это не имеет отношения к этому разговору.)
Вставляются следующие записи:
INSERT INTO Employees VALUES (1, NULL, 'Company President 1')
INSERT INTO Employees VALUES (2, NULL, 'Company President 2')
INSERT INTO Employees VALUES (3, 1, 'Company President 1 - VP')
INSERT INTO Employees VALUES (4, 2, 'Company President 2 - VP')
INSERT INTO Employees VALUES (5, 3, 'Company President 1 - VP - Secretary')
INSERT INTO Employees VALUES (6, 4, 'Company President 2 - VP - Secretary')
INSERT INTO Employees VALUES (7, 5, 'Company President 1 - VP - Secretary - Sandwich Delivery')
Эти вставки представляют собой:
Company President 1
Company President 1 - VP
Company President 1 - VP - Secretary
Company President 1 - VP - Secretary - Sandwich Delivery
Company President 2
Company President 2 - VP
Company President 2 - VP - Secretary
Я пытаюсь сделать для всех сотрудников, у которых есть NULL ParentEmployeeId
, я хочу найти последнего человека в цепочке, который в этом примере будет «Company President 1 - VP - Secretary - Sandwich Delivery
» и «Company President 2 - VP - Secretary
».
У меня есть следующий CTE, который дает мне все, включая уровень вложенности, но я не уверен, куда идти дальше. Я хотел бы избежать курсоров, если это возможно.
Кроме того, и это очень важно, у меня есть логика в другом месте, которая гарантирует, что у сотрудника может быть только 1 прямой подчиненный. Таким образом, хотя схема технически это допускает, Company President 1
никогда не будет иметь в списке двух вице-президентов.
WITH EmployeeRec(EmployeeId, ParentEmployeeId, Name, Level) AS
(
SELECT
EmployeeId,
ParentEmployeId,
Name,
1 as [Level]
FROM
Employees
WHERE
ParentEmployeId IS NULL
UNION ALL
SELECT
E.EmployeeId,
E.ParentEmployeId,
E.Name,
R.[Level] + 1
FROM
Employees E
INNER JOIN
EmployeeRec R
ON
E.ParentEmployeId = R.EmployeeId
)
SELECT * FROM EmployeeRec