如何在SQL Server中创建一个接受数据列的函数?

时间:2010-05-07 13:22:08

标签: sql sql-server tsql cursor user-defined-functions

我在本周早些时候在SQL Server 2008中创建了以下函数,该函数接受两个参数并使用它们来选择“详细”记录列,并将它们作为逗号分隔值的单个varchar列表返回。现在我开始思考它,我想采用这个表和特定于应用程序的函数,使其更通用。

我不熟悉定义SQL函数,因为这是我的第一个。如何更改此功能以接受单个“列”数据,以便我可以更通用的方式使用它?

而不是打电话:

SELECT ejc_concatFormDetails(formuid, categoryName)

我想让它的工作方式如下:

SELECT concatColumnValues(SELECT someColumn FROM SomeTable)

这是我的功能定义:

FUNCTION [DNet].[ejc_concatFormDetails](@formuid AS int, @category as VARCHAR(75))
RETURNS VARCHAR(1000) AS
BEGIN
 DECLARE @returnData VARCHAR(1000)
 DECLARE @currentData VARCHAR(75)
 DECLARE dataCursor CURSOR FAST_FORWARD FOR
  SELECT data FROM DNet.ejc_FormDetails WHERE formuid = @formuid AND category = @category

 SET @returnData = ''

 OPEN dataCursor

 FETCH NEXT FROM dataCursor INTO @currentData
 WHILE (@@FETCH_STATUS = 0)
 BEGIN
  SET @returnData = @returnData + ', ' + @currentData
  FETCH NEXT FROM dataCursor INTO @currentData
 END

 CLOSE dataCursor
 DEALLOCATE dataCursor

 RETURN SUBSTRING(@returnData,3,1000)
END

如您所见,我在函数中选择列数据,然后使用游标循环结果以构建逗号分隔的varchar。

如何更改此选项以接受作为结果集的单个参数,然后使用游标访问该结果集?

4 个答案:

答案 0 :(得分:5)

其他人已经回答了你的主要问题 - 但是让我指出你的功能的另一个问题 - 可怕的使用CURSOR!

你可以轻松重写这个函数,不使用游标,没有WHILE循环 - 没有那样。它会更快,也更容易 - 代码更少:

FUNCTION DNet.ejc_concatFormDetails
            (@formuid AS int, @category as VARCHAR(75))
RETURNS VARCHAR(1000) 
AS
    RETURN 
      SUBSTRING(
        (SELECT ', ' + data
         FROM DNet.ejc_FormDetails 
         WHERE formuid = @formuid AND category = @category
         FOR XML PATH('')
        ), 3, 1000)

诀窍是使用FOR XML PATH('') - 这会返回data列和固定', '分隔符的连续列表。在其上添加SUBSTRING()即可完成!就像那样简单.....没有顽固的慢CURSOR,没有混乱的连接和所有那些粘糊糊的代码 - 只有一个陈述而且就是这样。

答案 1 :(得分:2)

您可以使用表值参数:

CREATE FUNCTION MyFunction(
    @Data AS TABLE (
        Column1 int,
        Column2 nvarchar(50),
        Column3 datetime
    )
)
RETURNS NVARCHAR(MAX)
AS BEGIN
    /* here you can do what you want */
END

答案 2 :(得分:2)

您可以在SQL Server 2008中使用Table Valued Parameters,这样您就可以将TABLE变量作为参数传递。对此的限制和示例都在该链接文章中。

但是,我还要指出,使用游标很可能会对性能造成痛苦。 您不需要使用游标,因为您可以在1个SELECT语句中执行所有操作:

SELECT @MyCSVString = COALESCE(@MyCSVString + ', ', '') + data 
FROM DNet.ejc_FormDetails 
WHERE formuid = @formuid AND category = @category

无需光标

答案 3 :(得分:0)

你的问题有点不清楚。在您的第一个SQL语句中,您似乎正在尝试将列传递给函数,但是没有WHERE子句。在第二个SQL语句中,您传递了一组行(来自SELECT的结果)。您能提供一些样本数据和预期结果吗?

如果不完全了解您的目标,您可以考虑将参数更改为表变量。填充调用代码本地的表变量并将其传递给函数。您可以将其作为存储过程执行,但不需要函数。