Sql 需要递归调用表函数
我正在使用SQL Server 2014。我有一个table函数,它接受2个参数并返回2列。为了简单起见,让我们调用函数Get_GroupNumberSql 需要递归调用表函数,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
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循环完成了,我会把答案给后人