【问题标题】:SQL Server: How to get all child records given a parent id in a self referencing tableSQL Server:如何在自引用表中获取给定父 ID 的所有子记录
【发布时间】:2010-12-13 18:25:42
【问题描述】:

您好,我有一个引用自身的表,我需要能够从给定的父 ID 中选择父记录及其所有子记录。

我的表如下:

ID   | ParentID | Name         
-----------------------              
1      NULL       A
2      1          B-1
3      1          B-2
4      2          C-1
5      2          C-2

所以对于上面的例子,我希望能够传入一个值 1 并获取上面的所有记录。

到目前为止,我已经提出了以下递归表值函数,但它的行为不符合预期(仅返回第一条记录)。

CREATE FUNCTION [dbo].[SelectBranches]
(   
    @id INT
    ,@parentId INT
)
RETURNS @branchTable TABLE
(
    ID INT
    ,ParentID INT
    ,Name INT
)
AS
BEGIN 

    IF @branchId IS NOT NULL BEGIN

        INSERT INTO @branchTable
        SELECT 
            ID
            ,ParentID
            ,Name
        FROM
            tblLinkAdvertiserCity
        WHERE
            ID = @id

    END

    INSERT INTO @branchTable
    SELECT
        br.ID
        ,br.ParentID
        ,br.Name
    FROM
        @branchTable b
    CROSS APPLY
        dbo.SelectBranches(NULL, b.ParentID) br

    RETURN
END
GO

【问题讨论】:

  • +1 表示在此处提问之前尝试解决自己问题。

标签: sql sql-server-2005 recursion hierarchy


【解决方案1】:

你可以试试这个

DECLARE @Table TABLE(
        ID INT,
        ParentID INT,
        NAME VARCHAR(20)
)

INSERT INTO @Table (ID,ParentID,[NAME]) SELECT 1, NULL, 'A'
INSERT INTO @Table (ID,ParentID,[NAME]) SELECT 2, 1, 'B-1'
INSERT INTO @Table (ID,ParentID,[NAME]) SELECT 3, 1, 'B-2'
INSERT INTO @Table (ID,ParentID,[NAME]) SELECT 4, 2, 'C-1'
INSERT INTO @Table (ID,ParentID,[NAME]) SELECT 5, 2, 'C-2'


DECLARE @ID INT

SELECT @ID = 2

;WITH ret AS(
        SELECT  *
        FROM    @Table
        WHERE   ID = @ID
        UNION ALL
        SELECT  t.*
        FROM    @Table t INNER JOIN
                ret r ON t.ParentID = r.ID
)

SELECT  *
FROM    ret

【讨论】:

  • +1 CTE 非常适合递归,但它们对调用自己的次数确实有相对较小的限制。如果您的嵌套级别真的很深,那么请注意。我认为限制是 100。
  • @Robin Day:默认为 100,您可以通过在查询末尾附加“WITH MAXRECURSION number”来更改它。数字 0 表示没有限制。
  • MAXRECURSION 提示的值介于 0 和 32,767 之间
  • @astander 你是救生员。如果可以的话,我会给你大约 +100 票!
  • 有没有办法传递像Level/Deep 这样的附加参数并仅限于此?
【解决方案2】:

CTE 中的递归看起来有点贵,所以我编写了这个函数,它利用递归函数调用,但比 CTE 递归快得多。

CREATE FUNCTION [dbo].[Fn_GetSubCategories]
(
@p_ParentCategoryId INT
) RETURNS @ResultTable TABLE 
(   
    Id INT
)
AS
BEGIN
--Insert first level subcategories.
INSERT INTO @ResultTable 
SELECT Id FROM Category WHERE ParentCategoryId = @p_ParentCategoryId OR Id = @p_ParentCategoryId

DECLARE @Id INT
DECLARE @ParentCategory TABLE(Id INT)

DECLARE cur_categories CURSOR
LOCAL STATIC READ_ONLY FORWARD_ONLY FOR 
SELECT Id FROM Category WHERE ParentCategoryId = @p_ParentCategoryId and Id != @p_ParentCategoryId
OPEN cur_categories
IF @@CURSOR_ROWS > 0
     BEGIN 
     FETCH NEXT FROM cur_categories INTO @Id
     WHILE @@FETCH_STATUS = 0
     BEGIN
        --Insert remaining level sub categories.
        IF EXISTS(SELECT 1 FROM Category WHERE ParentCategoryId = @Id AND Id != @Id)
        BEGIN
            INSERT INTO @ResultTable
            SELECT DISTINCT C.Id from Fn_GetSubCategories(@Id) C INNER JOIN @ResultTable R ON C.Id != R.Id
        END

     FETCH NEXT FROM cur_categories INTO @Id
     END

     --Delete duplicate records
     ;WITH CTE AS
     (SELECT *,ROW_NUMBER() OVER (PARTITION BY Id ORDER BY Id) AS RN FROM @ResultTable)
     DELETE FROM CTE WHERE RN<>1

END
CLOSE cur_categories
DEALLOCATE cur_categories

RETURN

END

【讨论】:

    【解决方案3】:

    除非您使用 Oracle,否则您的表结构不适合所描述的问题。您尝试做的是获取层次结构(遍历树结构)。

    有一篇文章,More Trees & Hierarchies in SQL,描述了一种解决层级问题的方法。他基本上为每一行添加了一个描述层次结构的“血统”列。

    【讨论】:

    • 问题说他正在使用 SQL Server 2005 ;)
    • @Abtin 不幸的是,我正在使用旧系统,因此根本无法更改数据库架构:(
    猜你喜欢
    • 1970-01-01
    • 2013-04-08
    • 2021-02-12
    • 1970-01-01
    • 2020-06-06
    • 2015-01-30
    • 1970-01-01
    • 1970-01-01
    • 1970-01-01
    相关资源
    最近更新 更多