web-dev-qa-db-ja.com

SQLServerで再帰関数を呼び出す方法

私は次のようなテーブルを持っています

cat_id  Cat_Name    Main_Cat_Id

1       veg         null
2       main course 1
3       starter     1
4       Indian      2
5       mexican     2
6       tahi        3
7       chinese     3
8       nonveg      null
9       main course 8
10      indian      9
11      starter     8
12      tahi        11
13      chinese     11

Main_Cat_Id is cat_idそれが属する以前に追加されたカテゴリの)

このテーブルは、vegカテゴリに2つのサブカテゴリメインコーススターターがあり、で識別される)製品のカテゴリに使用されます。 main_cat_idそして、これらのサブカテゴリには、インドおよびメキシコのようなサブカテゴリがあります。

そして、この分類はユーザーによって異なります。彼はさらにサブカテゴリをインド、メキシコに追加して、任意のレベルの分類を行うことができます

今、私は野菜を取る場合のように、任意のノードのすべてのサブカテゴリを選択する必要があります私は選択する必要があります

(1)veg > (2)main course(1) > (4)indian(2)
                           > (5)mexican(2)
       > (3)starter(1)     > (6)thai(3)
                           > (7)chinese(3)

文字列を1,2,4,5,3,6,7として形成します

これを行うために、私は次のようにSQL関数を作成しました

CREATE FUNCTION [dbo].[GetSubCategory_TEST]
( @MainCategory int, @Category varchar(max))
RETURNS varchar(max)
AS
BEGIN
    IF EXISTS (SELECT Cat_Id FROM Category WHERE Main_Cat_Id=@MainCategory)
    BEGIN
        DECLARE @TEMP TABLE
        (
            CAT_ID INT
        )
        INSERT INTO @TEMP(CAT_ID) SELECT Cat_Id FROM Category WHERE Main_Cat_Id=@MainCategory
        DECLARE @TEMP_CAT_ID INT
        DECLARE CUR_CAT_ID CURSOR FOR SELECT CAT_ID FROM @TEMP
            OPEN CUR_CAT_ID
            WHILE 1 =1
                BEGIN
                FETCH NEXT FROM CUR_CAT_ID
                INTO  @TEMP_CAT_ID;
                IF @@FETCH_STATUS <> 0
                    SET @Category=@Category+','+ CONVERT(VARCHAR(50), @TEMP_CAT_ID)
                    SET @Category = [dbo].[GetSubCategory](@TEMP_CAT_ID,@Category)
                END
                CLOSE CUR_CAT_ID
                DEALLOCATE CUR_CAT_ID
    END
    return @Category
END 

しかし、この関数は実行を続け、目的の出力が得られません。plzで何が問題になっているのかわかりません。これを取得するのに役立ちます。

7
Rhushikesh

これを構築するのに再帰関数は必要ありません。再帰関数を使用できます [〜#〜] cte [〜#〜] そのために。

何かのようなもの

DECLARE @TABLE TABLE(
    cat_id INT,
    Cat_Name VARCHAR(50),
    Main_Cat_Id INT
)

INSERT INTO @TABLE SELECT 1,'veg',null
INSERT INTO @TABLE SELECT 2,'main course',1
INSERT INTO @TABLE SELECT 3,'starter',1
INSERT INTO @TABLE SELECT 4,'Indian',2
INSERT INTO @TABLE SELECT 5,'mexican',2
INSERT INTO @TABLE SELECT 6,'tahi',3
INSERT INTO @TABLE SELECT 7,'chinese',3
INSERT INTO @TABLE SELECT 8,'nonveg',null
INSERT INTO @TABLE SELECT 9,'main course',8
INSERT INTO @TABLE SELECT 10,'indian',9
INSERT INTO @TABLE SELECT 11,'starter',8
INSERT INTO @TABLE SELECT 12,'tahi',11
INSERT INTO @TABLE SELECT 13,'chinese',11

;WITH Recursives AS (
        SELECT  *,
                CAST(cat_id AS VARCHAR(MAX)) + '\' ID_Path
        FROM    @TABLE
        WHERE   Main_Cat_Id IS NULL
        UNION ALL
        SELECT  t.*,
        r.ID_Path + CAST(t.cat_id AS VARCHAR(MAX)) + '\'
        FROM    @TABLE t INNER JOIN
                Recursives r    ON  t.Main_Cat_Id = r.cat_id
)
SELECT  *
FROM    Recursives
9
Adriaan Stander

恥ずかしいのですが、@ astandersciptを使用して文字列の結果を出しました。

最初に私はあなたが与えたデータを作成しました。次に、必要な行を収集し、XMLを使用してすべてを1つの行に配置します(関数STUFFは最初のコンマを削除します)

DECLARE @TABLE TABLE(
    cat_id INT,
    Cat_Name VARCHAR(50),
    Main_Cat_Id INT
)

DECLARE @Collected TABLE(
    cat_id INT
)

INSERT INTO @TABLE SELECT 1,'veg',null
INSERT INTO @TABLE SELECT 2,'main course',1
INSERT INTO @TABLE SELECT 3,'starter',1
INSERT INTO @TABLE SELECT 4,'Indian',2
INSERT INTO @TABLE SELECT 5,'mexican',2
INSERT INTO @TABLE SELECT 6,'tahi',3
INSERT INTO @TABLE SELECT 7,'chinese',3
INSERT INTO @TABLE SELECT 8,'nonveg',null
INSERT INTO @TABLE SELECT 9,'main course',8
INSERT INTO @TABLE SELECT 10,'indian',9
INSERT INTO @TABLE SELECT 11,'starter',8
INSERT INTO @TABLE SELECT 12,'tahi',11
INSERT INTO @TABLE SELECT 13,'chinese',11
INSERT INTO @TABLE SELECT 14,'chinese',6

DECLARE @nodeID INT = 1;
DECLARE @result VARCHAR(MAX);

;WITH Recursives AS (
        SELECT cat_id, main_cat_id
        FROM @TABLE
        WHERE Cat_Id = @nodeID
        UNION ALL
        SELECT T.cat_id, T.main_cat_id
        FROM @TABLE AS T 
        INNER JOIN Recursives AS R 
            ON  t.Main_Cat_Id = r.cat_id
)
INSERT INTO @Collected
SELECT cat_id
FROM Recursives


SELECT @result = STUFF(
        (SELECT ',' + CAST( cat_id AS VARCHAR)
        FROM @Collected
        ORDER BY cat_id
        FOR XML PATH('')
        ), 1,1,'')

SELECT @result
0
Darka

1が1に等しくなくなるまでカーソルを動かし続けるように要求したため、カーソルは無限にループしています。

WHILE 1 =1

1=1は常にtrueであるため、ループが終了することはなく、どこからでも明示的にbreakを除外することはありません。

カーソルの例をいくつか調べてみるとよいでしょう。たとえば、Microsoft T-SQLドキュメントの this one です。それらは非常に定型的であり、主要な構文を大幅に変更する必要はほとんどありません。

カーソルをopenした後の標準的なアプローチは、最初のfetch nextを実行して最初の結果を取得し、次に@@FETCH_STATUS = 0(0は-を意味する)を条件としてwhileループを開くことです。 成功 )。

unsuccessfulカーソルフェッチ状態のみをカーソル内で探しているため:

IF @@FETCH_STATUS <> 0

@Categoryの設定は、カーソルがセットの最後の行を超えた場合にのみ行われます。これはまさにあなたが望まないことだと思います。

@Category変数は関数への入力パラメーターであるため、スコープについてもわかりません。私は通常、操作する関数内に新しい変数を作成しますが、これが実際に問題を引き起こすかどうかはわかりません。

より一般的には、ここで何を達成しようとしているのか完全には理解していませんが、 Adriaan Standerの回答 が示唆するように、カーソルを含む再帰関数はおそらく正しい方法ではありません。