SQL SERVER数据库开发之存储过程应用

可能有不少朋友使用SQL SERVER做开发也已经有段日子,但还没有或者很少在项目中使用存储过程,或许有些朋友认为根本没有必要使用存储过程等等。其实当你一个项目做完到了维 护阶段时,就会发现存储过程给我们带来了好处了,修改方便,不能去改我们的应用程序,只需要改存储过程的内容,而且还可以使我们的程序速度得到提高。

QUOTE:
SQL SERVER 联机丛书中的定义:
SQLServer目前正日益成为WindowNT操作系统上面最为重要的一种数据库管理系统,随着 SQLServer2000的推出,微软的这种数据库服务系统真正的实现了在WindowsNT/2000系列操作系统一统天下的局面,在微软的操作系统 上面,没有任何一种数据库系统能与之相抗衡,包括数据库领域的领头羊甲骨文公司的看家数据库系统Oracle在内。不可否认,SQL Server最大的 缺陷就是只能运行在微软自己的操作系统上面,这一点是SQLServer的致命伤口。但在另一方面却也成了最好的促进剂,促使SQLServer在自己仅 有的“土地”上面将自己的功能发挥到了极至,最大限度的利用了NT系列操作系统的各种潜能!作为SQLServer数据库系统中很重要的一个概念就是存储 过程,合理的使用存储过程,可以有效的提高程序的性能;并且将商业逻辑封装在数据库系统中的存储过程中,可以大大提高整个软件系统的可维护性,当你的商业 逻辑发生了改变的时候,不再需要修改并编译客户端应用程序以及重新分发他们到为数众多的用户手中,你只需要修改位于服务器端的实现相应商业逻辑的存储过程 即可。合理的编写自己需要的存储过程,可以最大限度的利用SQLServer的各种资源
  
存储过程是保存起来的可以接受和返回用户提供的参数的 Transact-SQL 语句的集合。
  可以创建一个过程供永久使用,或在一个会话中临时使用(局部临时过程),或在所有会话中临时使用(全局临时过程)。
  也可以创建在 Microsoft SQL Server 启动时自动运行的存储过程。


  要使用存储过程,首先我们必需熟悉一些基本的T-SQL语句,因为存储过程是由于一组T-SQL语句构成的,并且,我们需要了解一些关于函数、过程的概念,因为我们需要在应用程序中调用存储过程,就像我们调用应用程序的函数一样,不过调用的方法有些不同。

  下面我们来看一下存储过程的建立和使用方法。

一、创建存储过程

  和数据表一样,在使用之前我们需要创建存储过程,它的简明语法是:

QUOTE:
CREATE PROC 存储过程名称
    [参数列表(多个以“,”分隔)]
AS
SQL 语句


例:

QUOTE:
CREATE PROC upGetUserName
@intUserId        INT,
@ostrUserName NVARCHAR(20) OUTPUT                -- 要输出的参数
AS
BEGIN
        -- 将uName的值赋给 @ostrUserName 变量,即要输出的参数
        SELECT @ostrUserName=uName FROM uUser WHERE uId=@intUserId
END


   其中 CREATE PROC 语句(完整语句为CREATE PROCEDURE)的意思就是告诉SQL SERVER,现在需要建立一个存储过程,upGetUserName 就是存储过程名称,@intUserId 和 @ostrUserName 分别是该存储过程的两个参数,注意,在SQL SERVER中,所有用户定义的变量都以“@”开头,OUTPUT关键字表示这个参数是用来输出的,AS之 后就是存储过程内容了。只要将以上代码在“查询分析器”里执行一次,SQL SERVER就会在当前数据库中创建一个名为“upGetUserName”的存储过程。你可以打开“企业管理器”,选择当前操作的数据库,然后在左边的 树型列表中选择“存储过程”,此时就可以在右边的列表中看到你刚刚创建的存储过程了(如果没有,刷新一下即可)。

二、存储过程的调用

  之前我们已经创建了一个名为“upGetUserName”的存储过程,从字面理解该存储过程的功能是用来取得某一个用户的名称。存储过程建立好了,接下来就是要在应用程序里调用了,下面看一下在ASP程序里的调用。

QUOTE:
Dim adoComm
'// 创建一个对象,我们用来调用存储过程
Set adoComm = CreateObject("ADODB.Command")
With adoComm
        '// 设置连接,设 adoConn 为已经连接的 ADODB.Connection 对象
        .ActiveConnection = adoConn
        '// 类型为存储过程,adCmdStoredProc = 4
        .CommandType = 4
        '// 存储过程名称
        .CommandText = "upGetUserName"
        '// 设置用户编号
        .Parameters.Item("@intUserId").Value = 1
        '// 执行存储过程
        .Execute
       
        '// 取得从存储过程返回的用户名称
        Response.Write "用户名:" & .Parameters.Item("@ostrUserName").Value
End With
'// 释放对象
Set adoComm = Nothing


  通过以上两步,我们已经可以创建和使用简单的存储过程了。下面我们来看一个稍微复杂点的存储过程,以进一步了解存储过程的应用。

三、存储过程的实际应用

  用户登录在ASP项目中经常会使用到,相信很多朋友也都做过类似的系统,但使用存储过程来做验证朋友可能不多,那么我们就以它来做例子,写一个简单的用户登录验证的存储过程。

QUOTE:
CREATE PROC upUserLogin
@strLoginName        NVARCHAR(20),
@strLoginPwd        NVARCHAR(20),
@blnReturn                BIT OUTPUT
AS
-- 定义一个临时用来保存密码的变量
DECLARE @strPwd NVARCHAR(20)
BEGIN
        -- 从表中查询当前用户的密码,赋值给 @strPwd 变量,下面要对他进行比较
        SELECT @strPwd=uLoginPwd FROM uUser WHERE uLoginName=@strLoginName

        IF @strLoginPwd = @strPwd
                BEGIN
                        SET @blnReturn = 1
                        -- 更新用户最后登录时间
                        UPDATE uUser SET uLastLogin=GETDATE() WHERE uLoginName=@strLoginName
                END
        ELSE
                SET @blnReturn = 0
END


  用户登录的存储过程建立好了,现在在程序里试一下吧。注意,在一个区域内如果有多条语句时,必需使用BEGIN...END关键字。

QUOTE:
Dim adoComm
'// 创建一个对象,我们用来调用存储过程
Set adoComm = CreateObject("ADODB.Command")
With adoComm
        '// 设置连接,设 adoConn 为已经连接的 ADODB.Connection 对象
        .ActiveConnection = adoConn
        '// 类型为存储过程,adCmdStoredProc = 4
        .CommandType = 4
        '// 存储过程名称
        .CommandText = "upUserLogin"
        '// 设置登录名称
        .Parameters.Item("@strLoginName").Value = "admin"
        '// 设置登录密码
        .Parameters.Item("@strLoginPwd").Value = "123456"
        '// 执行存储过程
        .Execute
       
        '// 判断是否登录成功
        If .Parameters.Item("@blnReturn").Value = 1 Then
                Response.Write "恭喜你,登录成功!"
        Else
                Response.Write "不是吧,好像错了哦。。。"
        End If
End With
'// 释放对象
Set adoComm = Nothing


  通过以上的步骤,简单用户登录验证过程也做完了,现在只要把它整合到程序中就可以实现简单的用户登录验证了,关于其他细节就由你自己来处理了。
  上面介绍的两个存储过程都是只返回一个值的,下面我们来看一个返回一个记录集的存储过程。

QUOTE:
CREATE PROC upGetUserInfos
@intUserGroup        INT
AS
BEGIN
        -- 从数据库中抽取符合条件的数据
        SELECT uName,uGroup,uLastLogin FROM uUser WHERE uGroup=@intUserGroup
        -- 插入一列合计
        UNION
        SELECT '合计人数:',COUNT(uGroup),NULL FROM uUser WHERE uGroup=@intUserGroup
END


  现在我们来看一下ASP程序的调用。

QUOTE:
Dim adoComm
Dim adoRt
'// 创建一个对象,我们用来调用存储过程
Set adoComm = CreateObject("ADODB.Command")
Set adoRs = CreateObject("ADODB.Recordset")
With adoComm
        '// 设置连接,设 adoConn 为已经连接的 ADODB.Connection 对象
        .ActiveConnection = adoConn
        '// 类型为存储过程,adCmdStoredProc = 4
        .CommandType = 4
        '// 存储过程名称
        .CommandText = "upGetUserInfos"
        '// 设置用户组
        .Parameters.Item("@intUserGroup").Value = 1
        '// 执行存储过程,和以上几个例子不同,这里使用RecordSet的Open方法
        adoRs. Open adoComm
        '// 显示第一个值
        Response.write adoRs.Fields(0).Value
End With
'// 释放对象
Set adoRs = Nothing
Set adoComm = Nothing
      1、在你编写存储过程的时候,最好在有可能和系统关键字的地方使用“[”和“]”将他包围起来,以避免在移植过程中出现的运行错误问题。

  2、在存储过程中使用系统存储过程SP_Executesql的注意事项

  我们在编写自己的存储过程的时候,往往在很多的情况下,会使用到系统的存储过程SP_Execute。但是需要的注意的是,如果你在这个存储过 程的参数(一般是一段SQL语句)当中进行了临时Table的操作,那末对于调用者来说,这个临时Table是不可见的,也就是说你无法通过临时 Table来在调用者和被调用者之间传递值。解决的方法是使用全局临时Table,也就是“##”开头的Table。

  3、在存储过程中使用临时Table和游标的注意事项

  如果我们的商业逻辑比较复杂,在存储过程当中,就需要一些媒介作为中转台,这时候临时表j就发挥了作用,但是请务必记得在使用完之后,即时删除使用到的临时Table。

  而在存储过程当中想要依次遍历一个记录集的唯一方法就是使用系统游标,同样要注意的是,在使用完成之后及时关闭和销毁游标对象释放他用到的资 源。并且不在万不得已的情况下,不要随意使用游标,因为他会占用较多的系统资源,尤其是对于大并发量的情况下,很容易使得系统资源耗尽而崩溃。

  使用临时Table和游标各有利弊,在使用的过程中要适当的利用即可!

  4、在存储过程中调用外部的ActiveX DLL程序

  有些特殊的情况下,我们可能会需要调用外部的ActiveX DLL程序,这个时候就需要使用到系统的存储过程sp_OACreate以及其他 的相关系统存储过程,都是以sp_OA开头的存储过程,可以自由的在自己的存储过程当中调用ActiveX DLL的各种方法和属性。比如下面的例子:

  DECLARE @object int

  DECLARE @hr int

  DECLARE @property varchar(255)

  DECLARE @return varchar(255)

  DECLARE @src varchar(255), @desc varchar(255)

  -- 建立一个对象(SQLDMO.SQLServer).

  EXEC @hr = sp_OACreate 'SQLDMO.SQLServer',

  @object  OUT

  IF @hr <> 0

  BEGIN

  EXEC sp_OAGetErrorInfo @object, @src OUT, @desc  OUT

  SELECT hr=convert(varbinary(4),@hr),

  Source=@src, Description=@desc

  RETURN

  END

  -- 设置对象的属性.

  EXEC @hr = sp_OASetProperty @object, 'HostName', 'Gizmo'

  IF @hr <> 0

  BEGIN

  EXEC sp_OAGetErrorInfo @object, @src OUT, @desc  OUT

  SELECT hr=convert(varbinary(4),@hr),

  Source=@src, Description=@desc

  RETURN

  END

  -- 通过OUTPUT参数获取对象的属性值.

  EXEC @hr = sp_OAGetProperty @object, 'HostName', @property OUT

  IF @hr <> 0

  BEGIN

  EXEC sp_OAGetErrorInfo @object, @src OUT, @desc  OUT

  SELECT hr=convert(varbinary(4),@hr), Source=@src, Description=@desc

  RETURN

  END

  PRINT @property

  -- 调用对象的方法

  EXEC @hr = sp_OAMethod @object, 'Connect', NULL, 'my_server', 'my_login', 'my_password'

  IF @hr <> 0

  BEGIN

  EXEC sp_OAGetErrorInfo @object, @src OUT, @desc

  OUT

  SELECT hr=convert(varbinary(4),@hr), Source=@src, Description=@desc

  RETURN

  END

  -- 销毁已经创建的ActiveX对象

  EXEC @hr = sp_OADestroy @object

  IF @hr <> 0

  BEGIN

  EXEC sp_OAGetErrorInfo @object, @src OUT, @desc

  OUT

  SELECT hr=convert(varbinary(4),@hr), Source=@src, Description=@desc

  RETURN

  END

  5、在存储过程中使用数据库事务处理

  在很多的情况下,我们在存储过程中都会遇到需要同时操作多个表的情况,这时候就需要避免在操作的过程中由于以外而造成的数据的不一致性。这时候就需要将操作多个表的操作放入到事务中进行处理。

  但是需要注意的是,不能在事务中使用return语句强行退出,这样会引发事务的非正常错误,不能保证数据的一致性。

  并且,一旦将多个处理放入事务当中,系统的处理速度会有所降低,所以应当将频繁操作的多个可分割的处理过程放入到多个存储过程当中,这样会大大提高系统的响应速度,但是前提是不违背数据的一致性。

  看完了上面的这些编写SQL Server存储过程当中的技巧,相信对您或多或少会有些帮助,也希望通过上面的一些经验总结,可以使得您在应用SQL Server存储过程的时候,有意识的可以避免一些弯路。

评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包

打赏作者

a3676212

你的鼓励将是我创作的最大动力

¥1 ¥2 ¥4 ¥6 ¥10 ¥20
扫码支付:¥1
获取中
扫码支付

您的余额不足,请更换扫码支付或充值

打赏作者

实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值