Warning: file_get_contents(/data/phpspider/zhask/data//catemap/4/sql-server-2008/3.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181

Warning: file_get_contents(/data/phpspider/zhask/data//catemap/5/url/2.json): failed to open stream: No such file or directory in /data/phpspider/zhask/libs/function.php on line 167

Warning: Invalid argument supplied for foreach() in /data/phpspider/zhask/libs/tag.function.php on line 1116

Notice: Undefined index: in /data/phpspider/zhask/libs/function.php on line 180

Warning: array_chunk() expects parameter 1 to be array, null given in /data/phpspider/zhask/libs/function.php on line 181
Sql server 如何在SQLServer中调用递归函数_Sql Server_Sql Server 2008_Sql Function - Fatal编程技术网

Sql server 如何在SQLServer中调用递归函数

Sql server 如何在SQLServer中调用递归函数,sql-server,sql-server-2008,sql-function,Sql Server,Sql Server 2008,Sql Function,我有一张桌子如下 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

我有一张桌子如下

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是它所属的先前添加的类别的Cat_Id

本表用于蔬菜类产品的类别,其中蔬菜类有两个子类主菜和起菜,由主菜类别标识 这些子类别又有印度和墨西哥的子类别

这种分类依赖于用户;他还可以为印第安人、墨西哥人添加更多的子类别,这样他就可以进行任何级别的分类

现在,我必须选择任何节点的所有子类别,就像我要选择veg一样

(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 

但是这个函数一直在执行,没有给出所需的输出。我不知道到底出了什么问题。请帮我得到这个函数。你不需要递归函数来构建它,你可以使用递归函数来实现它

差不多

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

我很惭愧,但我使用@astander scit给出了字符串结果

首先,我创建了您提供的数据。 第二,我收集我需要的行 然后使用XML将所有内容放在一行函数中,删除第一个逗号

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

您的光标无限循环,因为您要求它一直循环,直到1不再等于1:

而1=1

1=1始终为真,因此循环永远不会结束,并且不会在任何地方显式中断

您最好学习一些游标示例,例如在Microsoft T-SQL文档中。它们是相当公式化的,主要语法很少需要太多变化

打开游标后的标准方法是在获取第一个结果的旁边进行初始提取,然后在@fetch\u STATUS=0的条件下打开while循环

因为您只在游标内查找未成功的游标获取状态:

如果@@FETCH\u状态为0

@Category的设置仅在光标经过集合中的最后一行时发生。我怀疑这正是你不想要的

我也不确定@Category变量的作用域,因为它是函数的输入参数;我通常会在函数中创建新的变量来处理,但我不确定这是否会产生问题


更一般地说,虽然我不完全理解您在这里想要实现什么,但正如建议的那样,涉及游标的递归函数可能不是正确的方法

而1=1??它会无限的?类别的可能重复项不是固定的。这些类别和类别的级别不是固定的。@Rhushikesh:请阅读答案!您混淆了将示例与所提出问题的后续解决方案相匹配所需的示例数据构造。