一、示例

1.创建标量函数,返回单个值

--创建函数
create function FuncTest2(@id int=1)--函数名、参数列表(默认值 1)
returns nvarchar(50)--返回值类型,单个值
as
begin
declare @name nvarchar(50)
select @name=name from a_test where ID=@id
return (@name)
end

--执行函数,需要加入架构前缀,例:dbo
select dbo.FuncTest2(2)

  

2.创建内嵌表值函数,返回一个表

create function FuncTest(@id int)
returns table--返回值类型,数据集
as
begin
return (select * from a_test where ID=@id)
end
--执行函数,返回数据集作为一张表
select * from FuncTest(2)

  

3.创建多声明表值型用户自定义函数,其语法如下:

create function [owner_name.]function_name
([@parameter_name scalar_parameter_data_type[=default]][,...n])
returns @return_variable table <table_type_definition>
[with<function_option>[,...n]]
[as]
begin
function_body
return
end

<table_type_definition>::=
({column_definition|table_constraint}[,...n])

  

示例:

create function FuncTest3()
returns @tb table
(
name nvarchar(50)
)
as
begin
insert @tb select name from a_test
return
end
go
--使用
select * from dbo.FuncTest3()

  

各参数说明如下:
@return_variable :一个TABLE 类型的变量,用于存储和累积返回的表中的数据行。
其余参数与标量型用户自定义函数相同。
在多声明表值型用户自定义函数的函数体中允许使用下列Transact-SQL 语句:
赋值语句(Assignment statements);
流程控制语句(Control-of-Flow statements);
定义作用范围在函数内的变量和 游标的DECLARE 语句;
SELECT 语句;
编辑函数中定义的表变量的INSERT、 UPDATE 和DELETE 语句;
在函数中允许涉及诸如声明游 标、打开游标、关闭游标、释放游标这样的游标操作,对于读取游标而言,除非在FETCH 语句中使用INTO 从句来对某一变量赋值,否则不允许在函数中使用FETCH 语句来向客户端返回数据。
此外不确定性函数(Non-deterministic functions)不能在用户自定义函数中使用。
所谓不确定性函数是指那些使用相同的调用参数在不同时刻调用得到的返回值不同的函数。
这些函数如表下所示(全局变量也可以视为一种函数)。

 

二、sql函数和存储过程的区别

1.能够在select等SQL语句中直接使用自定义函数,存储过程不行。
2.自定义函数可以调用其他函数,也可以调用自己(递归)。
3.可以在表列和 CHECK 约束中使用自定义函数来实现特殊列或约束。
4.自定义函数不能有任何副作用。函数副作用是指对具有函数外作用域(例如数据库表的修改)的资源状态的任何永久性更改。

函数中的语句唯一能做的更改是对函数上的局部对象(如局部游标或局部变量)的更改。

不能在函数中执行的操作包括:对数据库表的修改,对不在函数上的局部游标进行操作,发送电子邮件,尝试修改 目录,以及生成返回至用户的结果集;存储过程没有此限制。
5.函数只能返回一个变量,而存储过程可以返回多个。

6.函数不支持输出参数,存储过程可以。(output)

 

三、描述

根据函数返回值形式的不同将用户自定义函数分为三种类型:标量函数(Scalar Function)、内嵌表值函数(Inline Function)、多声明表值函数(Multi-Statement Function)

标量函数:标量函数是对单一值操作,返回单一值。能够使用表达式的地方,就可以使用标量函数。像我们经常使用的left、getdate等,都属于标量函数。系统函数中的标量函数包括:数学函数、日期和时间函数、字符串函数、数据类型转换函数等。
内嵌表值函数:内嵌表值函数的功能相当于一个参数化的视图。它返回的是一个表,内联表值型函数没有由BEGIN-END 语句括起来的函数体。其返回的表由一个位于RETURN 子句中的SELECT 命令段从数据库中筛选出来。
多声明表值函数:可以看作标量型和内嵌表值型函数的结合体。它的返回值是一个表,但它和标量型函数一样有一个用BEGIN-END 语句括起来的函数体,返回值的表中的数据是由函数体中的语句插入的。由此可见,它可以进行多次查询,对数据进行多次筛选与合并,弥补了内联表值型函数的不足。

适用范围:
1.只查询,不修改数据库的状态(修改、删除表中记录等)
2.结果集需要通过递归等方法得到时,可以使用函数,函数比较灵活
3.结果集需要直接被引用时,可以使用函数。需要对结果集进行再加工(指放在select语句中等),可以使用函数,函数可以嵌在select等sql语句中。
注意事项:
用户自定义函数不能用于执行一系列改变数据库状态的操作,在编写自定义函数时需要注意的:
对于标量函数:
1.所有的入参前都必须加@
2.create后的返回,单词是returns,而不是return
3.returns后面的跟的不是变量,而是返回值的类型,如:int,char等。
4.在begin/end语句块中,是return。
内嵌表值函数:
1.只能返回table,所以returns后面一定是TABLE
2.AS后没有begin/end,只有一个return语句来返回特定的记录。
多语句表值函数:
1.returns后面直接定义返回的表类型,首先是定义表名,表明前面要加@,然后是关键字TABLE,最后是表的结构。
2.在begin/end语句块中,直接将需要返回的结果insert到returns定义的表中就可以了,在最后return时,会将结果返回。
3.最后只需要return,return后面不跟任何变量。

疑问:自定义函数不能修改数据库,但它可以调用存储过程,那么在自定义函数中调用一个有修改数据库的操作的存储过程,这个自定义函数能不能执行?
答:自定义函数只能调用扩展存储过程,但是SQL Server 2008的后续版本将删除该功能,不再支持扩展存储过程,所以应避免在开发中使用扩展存储过程。

因此,可以得出结论是:实际开发中,函数不会去调用存储过程,也就无法对数据库进行修改操作了。

 

posted on 2016-04-05 14:18  邢帅杰  阅读(310)  评论(0编辑  收藏  举报