Sql 需要递归调用表函数

Sql 需要递归调用表函数,sql,sql-server,sql-server-2014,Sql,Sql Server,Sql Server 2014,我正在使用SQL Server 2014。我有一个table函数,它接受2个参数并返回2列。为了简单起见,让我们调用函数Get_GroupNumber Get_GroupNumber('Smith', '123-456-789') 这将返回2列。再次保持简单,假设它是Reason1和Reason2 Reason1 = '111-222-333' Reason2 = '555'666'777' 我需要将返回的Reason1传递回Get_GroupNumber,并继续这样做,直到它在Reason

我正在使用SQL Server 2014。我有一个table函数,它接受2个参数并返回2列。为了简单起见,让我们调用函数Get_GroupNumber

Get_GroupNumber('Smith', '123-456-789')
这将返回2列。再次保持简单,假设它是Reason1和Reason2

Reason1 = '111-222-333'
Reason2 = '555'666'777'
我需要将返回的Reason1传递回
Get_GroupNumber
,并继续这样做,直到它在Reason1中返回NULL,在Reason2中也返回NULL

因此,第二个调用将如下所示:

Get_GroupNumber('Smith','111-222-333')

如果您不想使用迭代方法(例如,使用
while
循环),那么递归CTE可以完成这项工作。您可以阅读有关递归CTE的更多信息,或者查看以下示例:

-- Here's a "dummy" function that has some hard-coded return values for illustrative
-- purposes. It will return ('111-222-333', '555-666-777') when it gets 123-456-789
-- as its second parameter, ('999-999-999', '000-000-000') when it gets 111-222-333
-- as its second parameter, and (null, null) otherwise.
create function Get_GroupNumber
(   
    @Param1 varchar(32),
    @Param2 varchar(32)
)
returns table
as
return 
(
    select
        Reason1 = case @Param2 when '123-456-789' then '111-222-333' when '111-222-333' then '999-999-999' else null end,
        Reason2 = case @Param2 when '123-456-789' then '555-666-777' when '111-222-333' then '000-000-000' else null end
);
go

-- The sample inputs from your question.
declare @Param1 varchar(32) = 'Smith';
declare @Param2 varchar(32) = '123-456-789';

-- And finally, the good stuff:
with GroupNumberCTE as
(
    -- Base case: pass the original parameters, get the results from the function, and
    -- create a new field called "Call #" that will illustrate the order in which the
    -- various calls to Get_GroupNumber were made. (This field is purely informational;
    -- you can remove it if you don't want it.)
    select 
        [Call #] = 1,
        Reason1, 
        Reason2 
    from 
        Get_GroupNumber(@Param1, @Param2)

    union all

    -- Recursive case: if the previous call returned a non-null value in either field,
    -- invoke the function again with the original @Param1 and the Reason1 from the
    -- previous call. 
    select
        [Call #] = [Previous].[Call #] + 1,
        [Next].Reason1,
        [Next].Reason2
    from
        GroupNumberCTE [Previous]
        cross apply Get_GroupNumber(@Param1, [Previous].Reason1) [Next]
    where
        [Previous].Reason1 is not null or
        [Previous].Reason2 is not null
)
select * from GroupNumberCTE;
结果集:

Call #   Reason1       Reason2
----------------------------------
1        111-222-333   555-666-777
2        999-999-999   000-000-000
3        NULL          NULL
我应该指出这里有一个危险:我的CTE本身的结构中没有任何东西可以保证递归最终会结束。因此,您必须确保对于任何可行的初始输入集,
Get_GroupNumber
的实现最终将返回两个空值。只要是这样,这种方法就应该很有效

Using While Loop

DECLARE @key varchar(max);
DECLARE @retVal varchar(max);
SET @key = '123-456-789'
While @Key is not null
BEGIN
    set @retval = @key
    SET @key = (SELECT Return2 from [dbo].[Get_GroupNumber]('Smith',@key))
END;

print @retVal

我知道我只使用Return2列的返回值在这里进行测试

1。函数是否返回null?2.为什么不能使用
While
循环?对我来说,这听起来可能也是一个无止境的循环。我无法编辑函数本身。是的,函数可以返回NULL。虽然循环听起来不错,但我将尝试查找如何做到这一点(对不起,SQL编程新手)Ok。我说的是调用函数(如果有的话)的存储过程。在其中放置一个循环。是的,我将尝试为此创建一个SP。谢谢你,我们也用while循环完成了,我会把答案给后人