控件中国网现已改版,您看到的是老版本网站的镜像,系统正在为您跳转到新网站首页,请稍候.......
中国最专业的商业控件资讯网产品咨询电话:023-67870900 023-67871946
产品咨询EMAIL:SALES@COMPONENTCN.COM

表值函数

作者:佚名 出处:互联网 2010年12月27日 阅读:

表值函数
表值函数提供强大的结果集生成能力。它可以在查询内部表或视图允许的任何地方使用。表值函数在使用上比返回一个结果集的存储过程更灵活,因为函数的结果集可以联接到查询中的其他表。

SQL Server中有两种表值函数。内联表值函数在概念上与带参数的视图类似。多语句表值函数允许多条语句在表变量中创建结果集来返回。

1. 内联表值函数

创建内联表值函数很简单。内联表值函数的内容是一条带参数的SELECT语句。返回数据类型永远是表,不过返回表的结构由SELECT语句的结构来定义。下面是内联表值函数的一个例子,检索给定CustomerID的商品销售总量。

USE AdventureWorks2008;  GO  CREATE FUNCTION Sales.ufnSalesByCustomer (@CustomerID int)  RETURNS TABLE  AS  RETURN  (   SELECT P.ProductID, P.Name, SUM(SD.LineTotal) AS Total   FROM Production.Product AS P      JOIN Sales.SalesOrderDetail AS SD       ON SD.ProductID = P.ProductID      JOIN Sales.SalesOrderHeader AS SH      ON SH.SalesOrderID = SD.SalesOrderID   WHERE SH.CustomerID = @CustomerID   GROUP BY P.ProductID, P.Name  );  GO 注意,函数体由一条RETURN语句组成。使用这个函数的一个例子如下所示:

SELECT * FROM Sales.ufnSalesByCustomer(30052); 内联表值函数功能强大,在要求参数化查询的情况下值得考虑。它们在结果集如何使用上提供更多的灵活性。

2. 多语句表值函数

多语句表值函数允许多条语句来创建表的内容。多语句表值函数可以用来替换使用多个步骤来构建结果集的存储过程。

多语句表值函数允许开发人员使用多个步骤动态地填充表,这一点与存储过程类似,不过它们可以在SELECT语句中像表那样被引用。

使用多语句表值函数时,表的结构必须在函数头定义。要为表使用一个变量名,并且所有修改数据的操作只能引用表变量。

下面的例子是一个函数,类似上一节中创建的ufnSalesByCustomer。首先创建表变量,然后使用刚才创建的标量函数来更新表变量,让它包含总的存货清单。创建函数的语句如下所示:

USE AdventureWorks2008;  GO  CREATE FUNCTION Sales.ufnSalesByCustomerMS (@CustomerID int)  RETURNS @table TABLE  ( ProductID int PRIMARY KEY NOT NULL,   ProductName nvarchar(50) NOT NULL,   TotalSales numeric(38,6) NOT NULL,   TotalInventory int NOT NULL )  AS  BEGIN   INSERT INTO @table   SELECT P.ProductID, P.Name, SUM(SD.LineTotal) AS Total, 0   FROM Production.Product AS P   JOIN Sales.SalesOrderDetail SD ON SD.ProductID = P.ProductID   JOIN Sales.SalesOrderHeader SH ON SH.SalesOrderID = SD.SalesOrderID   WHERE SH.CustomerID = @CustomerID   GROUP BY P.ProductID, P.Name;    UPDATE @table   SET TotalInventory = dbo.ufnGetTotalInventoryStock(ProductID);    RETURN;  END; 执行这个函数与执行前面的内联函数一样:

SELECT * FROM Sales. ufnSalesByCustomerMS (30052);

-------------------------------------------------------------------------------------

表值函数和标量值函数的不同是 表值函数是返回一个Table类型 Table类型相当与一张存储在内存中的一张虚拟表.

表值函数的语法:

CREATE FUNCTION [ schema_name. ] function_name
( [ { @parameter_name [ AS ] [ type_schema_name. ] parameter_data_type
    [ = default ] }
    [ ,...n ]
]
)
RETURNS TABLE
    [ WITH <function_option> [ ,...n ] ]
    [ AS ]
    RETURN [ ( ] select_stmt [ ) ]
[ ; ]

现在来写一个比较实用的表值函数..

写一个切割字符串的表值函数

view plaincopy to clipboardprint?
--------------------------这个函数用来切割字符串的-----------------  
--函数的参数 第一个是要切割的字符串 第二个是要以什么字符串切割  
CREATE FUNCTION Split(@Text NVARCHAR(4000),@Sign NVARCHAR(4000))  
RETURNS @tempTable TABLE(id INT IDENTITY(1,1) PRIMARY KEY,[VALUE] NVARCHAR(4000))  
AS  
BEGIN  
    DECLARE @StartIndex INT                --开始查找的位置  
    DECLARE @FindIndex INT                --找到的位置  
    DECLARE @Content    VARCHAR(4000)    --找到的值  
    --初始化一些变量  
    SET @StartIndex = 1 --T-SQL中字符串的查找位置是从1开始的  
    SET @FindIndex=0  
      
    --开始循环查找字符串逗号  
    WHILE(@StartIndex <= LEN(@Text))  
    BEGIN  
        --查找字符串函数 CHARINDEX 第一个参数是要找的字符串  
        --                            第二个参数是在哪里查找这个字符串  
        --                            第三个参数是开始查找的位置  
        --返回值是找到字符串的位置  
        SELECT @FindIndex = CHARINDEX(@Sign,@Text,@StartIndex)  
        --判断有没找到 没找到返回0  
        IF(@FindIndex =0 OR @FindIndex IS NULL)  
        BEGIN  
            --如果没有找到者表示找完了  
            SET @FindIndex = LEN(@Text)+1  
        END  
        --截取字符串函数 SUBSTRING 第一个参数是要截取的字符串  
        --                            第二个参数是开始的位置  
        --                            第三个参数是截取的长度  
        --@FindIndex-@StartIndex 表示找的的位置-开始找的位置=要截取的长度  
        --LTRIM 和 RTRIM 是去除字符串左边和右边的空格函数  
        SET @Content = LTRIM(RTRIM(SUBSTRING(@Text,@StartIndex,@FindIndex-@StartIndex)))  
        --初始化下次查找的位置  
        SET @StartIndex = @FindIndex+1  
        --把找的的值插入到要返回的Table类型中  
        INSERT INTO @tempTable ([VALUE]) VALUES (@Content)   
    END  
    RETURN  
END
--------------------------这个函数用来切割字符串的-----------------
--函数的参数 第一个是要切割的字符串 第二个是要以什么字符串切割
CREATE FUNCTION Split(@Text NVARCHAR(4000),@Sign NVARCHAR(4000))
RETURNS @tempTable TABLE(id INT IDENTITY(1,1) PRIMARY KEY,[VALUE] NVARCHAR(4000))
AS
BEGIN
     DECLARE @StartIndex INT                --开始查找的位置
     DECLARE @FindIndex INT                --找到的位置
     DECLARE @Content    VARCHAR(4000)    --找到的值
     --初始化一些变量
     SET @StartIndex = 1 --T-SQL中字符串的查找位置是从1开始的
     SET @FindIndex=0
    
     --开始循环查找字符串逗号
     WHILE(@StartIndex <= LEN(@Text))
     BEGIN
         --查找字符串函数 CHARINDEX 第一个参数是要找的字符串
         --                            第二个参数是在哪里查找这个字符串
         --                            第三个参数是开始查找的位置
         --返回值是找到字符串的位置
         SELECT @FindIndex = CHARINDEX(@Sign,@Text,@StartIndex)
         --判断有没找到 没找到返回0
         IF(@FindIndex =0 OR @FindIndex IS NULL)
         BEGIN
             --如果没有找到者表示找完了
             SET @FindIndex = LEN(@Text)+1
         END
         --截取字符串函数 SUBSTRING 第一个参数是要截取的字符串
         --                            第二个参数是开始的位置
         --                            第三个参数是截取的长度
         --@FindIndex-@StartIndex 表示找的的位置-开始找的位置=要截取的长度
         --LTRIM 和 RTRIM 是去除字符串左边和右边的空格函数
         SET @Content = LTRIM(RTRIM(SUBSTRING(@Text,@StartIndex,@FindIndex-@StartIndex)))
         --初始化下次查找的位置
         SET @StartIndex = @FindIndex+1
         --把找的的值插入到要返回的Table类型中
         INSERT INTO @tempTable ([VALUE]) VALUES (@Content)
     END
     RETURN
END

这个函数的作用就是类似.Net中的string类的Split方法

现在来测试这个函数

这个函数返回的是Table类型   所以可以用下面的语法来调用

view plaincopy to clipboardprint?
SELECT * FROM dbo.Split('a,b,c,d,e,f,g',',')
SELECT * FROM dbo.Split('a,b,c,d,e,f,g',',')

得到的结果

热推产品

  • ActiveReport... 强大的.NET报表设计、浏览、打印、转换控件,可以同时用于WindowsForms谀坔攀戀Forms平台下......
  • AnyChart AnyChart使你可以创建出绚丽的交互式的Flash和HTML5的图表和仪表控件。可以用于仪表盘的创......
首页 | 新闻中心 | 产品中心 | 技术文档 | 友情连接 | 关于磐岩 | 技术支持中心 | 联系我们 | 帮助中心 Copyright-2006 ComponentCN.com all rights reserved.重庆磐岩科技有限公司(控件中国网) 版权所有 电话:023 - 67870900 传真:023 - 67870270 产品咨询:sales@componentcn.com 渝ICP备12000264号 法律顾问:元炳律师事务所 重庆市江北区塔坪36号维丰创意绿苑A座28-5 邮编:400020
在线客服
在线客服系统
在线客服
在线客服系统