Рекурсивный вид запроса T-SQL - PullRequest
1 голос
/ 17 июня 2010

На основании следующей таблицы

ID    Path       
---------------------------------------
1  \\Root
2  \\Root\Node0
3  \\Root\Node0\Node1
4  \\Root\Node0\Node2
5  \\Root\Node3
6  \\Root\Node3\Node4
7  \\Root\Node5
...
N  \\Root\Node5\Node6\Node7\Node8\Node9\Node10

и так далее ...

В этой таблице около 1000 строк. Я хочу отображать отдельные узлы в отдельных столбцах. Максимальное количество отображаемых столбцов - 5 (т. Е. От узла до 5 уровня). Итак, вывод будет выглядеть как показано ниже

ID    Path           Level 0   Level 1  Level 2  Level 3  Level 4  Level 5
----------------------------------------------------------------------------------------
1  \\Root                    Root      Null     Null     Null     Null     Null
2  \\Root\Node0              Root      Node 0   Null     Null     Null     Null
3  \\Root\Node0\Node1        Root      Node 0   Node 1   Null     Null     Null
4  \\Root\Node0\Node2        Root      Node 0   Node 2   Null     Null     Null
5  \\Root\Node3              Root      Node 3   Null     Null     Null     Null
6  \\Root\Node3\Node4        Root      Node 3   Node 4   Null     Null     Null
7  \\Root\Node5              Root      Node 5   Null     Null     Null     Null
...
N  (see in above table)      Root      Node 5   Node 6   Node 7   Node 8 Node 9

Единственный способ, которым я могу придумать, - это открыть курсор, перебрать каждую строку и выполнить разбиение строки, просто извлечь первые 5 узлов и затем вставить их во временную таблицу.

Пожалуйста, предложите.

Спасибо

1 Ответ

0 голосов
/ 17 июня 2010

Вам нужна табличная функция разбиения, похожая на следующую:

CREATE FUNCTION [dbo].[udf_Split] (@DelimitedList nvarchar(max), @Delimiter nvarchar(2) = ',')
RETURNS @SplitResults TABLE (Position int NOT NULL PRIMARY KEY, Value nvarchar(max))
AS
Begin
    Declare @DelimiterLength int
    Set @DelimiterLength = DataLength(@Delimiter) / 2

    If Left(@DelimitedList, @DelimiterLength) <> @Delimiter
        Set @DelimitedList = @Delimiter + @DelimitedList

    If Right(@DelimitedList, @DelimiterLength) <> @Delimiter
        Set @DelimitedList = @DelimitedList + @Delimiter

    Insert @SplitResults(Position, Value)
    Select CharIndex(@Delimiter, A.list, N.Value) + @DelimiterLength            
        , Substring (
                    A.List
                    , CharIndex(@Delimiter, A.list, N.Value) + @DelimiterLength         
                    , CharIndex(@Delimiter, A.list, N.Value + 1)                            
                        - ( CharIndex(@Delimiter, A.list, N.Value) + @DelimiterLength ) 
                    )
    From dbo.Numbers As N
        Cross Join (Select @DelimitedList As list) As A
    Where N.Value > 0
        And N.Value < LEN(A.list)
        And Substring(A.list, N.Value, @DelimiterLength) = @Delimiter
    Order By N.Value

    Return
End

Эта функция основана на существовании таблицы Numbers, которая содержит последовательный список целых чисел.Теперь вы можете взять исходные данные и сделать что-то вроде:

With TableData As
    (
    Select 1 As Id, '\\Root' As [Path]
    Union Select All 2, '\\Root\Node0'
    Union Select All 3, '\\Root\Node0\Node1'
    Union Select All 4, '\\Root\Node0\Node2'
    Union Select All 5, '\\Root\Node3'
    Union Select All 6, '\\Root\Node3\Node4'
    Union Select All 7, '\\Root\Node5'
    )
    , SplitData As
    (
    Select T.Id, T.[Path], S.Value
        , Row_Number() Over ( Partition By T.Id Order By S.Position ) As Level
    From TableData As T
        Cross Apply dbo.udf_Split( (Substring(T.[Path],2,Len(T.[Path])) + '\') , '\') As S
    )
Select Id, [Path]
    , Min( Case When Level = 1 Then S.Value End ) As Level0
    , Min( Case When Level = 2 Then S.Value End ) As Level1
    , Min( Case When Level = 3 Then S.Value End ) As Level2
    , Min( Case When Level = 4 Then S.Value End ) As Level3
    , Min( Case When Level = 5 Then S.Value End ) As Level4
From SplitData As S
Group By Id, [Path]
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...