SQL存储过程入门

 

一、SQL存储过程的概念,优点及语法?
  整理在学习程序过程之前,先了解下什么是存储过程?为什么要用存储过程,他有那些优点

  定义:将常用的或很复杂的工作,预先用SQL语句写好并用一个指定的名称存储起来, 那么以后要叫数据库提供与已定义好的存储过程的功能相同的服务时,只需调用execute,即可自动完成命令。

  讲到这里,可能有人要问:这么说存储过程就是一堆SQL语句而已啊? Microsoft公司为什么还要添加这个技术呢?

  那么存储过程与一般的SQL语句有什么区别呢?

  存储过程的优点:

  1.存储过程只在创造时进行编译,以后每次执行存储过程都不需再重新编译,而一般SQL语句每执行一次就编译一次,所以使用存储过程可提高数据库执行速度。

  2.当对数据库进行复杂操作时(如对多个表进行Update,Insert,Query,Delete时),可将此复杂操作用存储过程封装起来与数据库提供的事务处理结合一起使用。

  3.存储过程可以重复使用,可减少数据库开发人员的工作量

  4.安全性高,可设定只有某此用户才具有对指定存储过程的使用权

  存储过程的种类:

  1.系统存储过程:以sp_开头,用来进行系统的各项设定.取得信息.相关管理工作,

  如 sp_help就是取得指定对象的相关信息

  2.扩展存储过程 以XP_开头,用来调用操作系统提供的功能

  以下为引用的内容:

  exec master..xp_cmdshell ‘ping 10.8.16.1′

  3.用户自定义的存储过程,这是我们所指的存储过程

  常用格式

  以下为引用的内容:

  Create procedure procedue_name

  [@parameter data_type][output]

  [with]{recompileencryption}

  as

  sql_statement

解释:

  output:表示此参数是可传回的

  with {recompileencryption}

  recompile:表示每次执行此存储过程时都重新编译一次

  encryption:所创建的存储过程的内容会被加密

  二、SQL存储过程学习:存储过程的创建

  表book的内容如下

  编号     书名      价格

  001    C语言入门     $30

  002  PowerBuilder报表开发? $52

  实例1:查询表Book的内容的存储过程

  create proc query_book?
  as?
  select * from book?
  go?
  exec query_book?
  实例2:加入一笔记录到表book,并查询此表中所有书籍的总金额

  Create proc insert_book?
  @param1 char(10),@param2 varchar(20),@param3 money,@param4 money output?
  with encryption ———加密?
  as?
  insert book(编号,书名,价格) Values(@param1,@param2,@param3)?
  select @param4=sum(价格) from book?
  go?
执行例子:?
  declare @total_price money?
  exec insert_book ’003′,’Delphi 控件开发指南’,$100,@total_price?
  print ‘总金额为’+convert(varchar,@total_price)?
  go?
  存储过程的3种传回值:

  1.以Return传回整数

  2.以output格式传回参数

  3.Recordset

  传回值的区别:

  output和return都可在批次程式中用变量接收,而recordset则传回到执行批次的客户端中

  实例3:设有两个表为Product,Order,其表内容如下:

以下为引用的内容:

  Product

  产品编号 产品名称 客户订数

  001    钢笔   30

  002    毛笔    50

  003????????? 铅笔?   100

  order

  产品编号 客户名 客户订金

  001   南山区  $30

  002   罗湖区  $50

  003   宝安区  $4

  请实现按编号为连接条件,将两个表连接成一个临时表,该表只含编号.产品名.客户名.订金.总金额,

?
  总金额=订金*订数,临时表放在存储过程中
  代码如下:
  Create proc temp_sale?
  as?
  select a.产品编号,a.产品名称,b.客户名,b.客户订金,a.客户订数* b.客户订金 as总金额?
  into #temptable from Product a inner join order b on a.产品编号=b.产品编号?
  if @@error=0?
  print ‘Good’?
  else?
  print ‘Fail’?
  go?
  三、SQL存储过程学习:存储过程的调用

  调用带参数存储过程的几种方式

  1) 这也是最简单的方法,两个输入参数,无返回值,用于Insert,Update,Delete操作较多。

  以下为引用的内容:

  conn.Execute “procname varvalue1,varvalue2″

  2) 如果要返回 Recordset 集:

  以下为引用的内容:

  set rs = server.createobject(“adodb.recordset”)

  rs.Open “Exec procname varvalue1, varvalue2″,conn

  3) 以上两种方法都不能有返回值,(Recordset除外),如果要得到返回值,需要用Command的方法。

  首先说明,返回值有两种。一种是在存储过程中直接return一个值,就象C和VB的函数返回值那样;另一种是可以返回多个值,存储这些值的变量名称需要在调用参数中先行指定。

  四、特殊的存储过程-触发器

  1.触发器的概念及作用

  触发器是一种特殊类型的存储过程,它不同于我们前面介绍过的存储过程。触发器主要是通过事件进行触发而被执行的,而存储过程可以通过存储过程名字而被直接调用。当对某一表进行诸如Update、 Insert、 Delete 这些操作时,SQL Server 就会自动执行触发器所定义的SQL 语句,从而确保对数据的处理必须符合由这些SQL 语句所定义的规则。
  触发器的主要作用就是其能够实现由主键和外键所不能保证的复杂的参照完整性和数据的一致性。除此之外,触发器还有其它许多不同的功能:?
  (1) 强化约束(Enforce restriction)

  触发器能够实现比CHECK 语句更为复杂的约束。

  (2) 跟踪变化Auditing changes

  触发器可以侦测数据库内的操作,从而不允许数据库中未经许可的指定更新和变化。

  (3) 级联运行(Cascaded operation)。

  触发器可以侦测数据库内的操作,并自动地级联影响整个数据库的各项内容。例如,某个表上的触发器中包含有对另外一个表的数据操作(如删除,更新,插入)而该操作又导致该表上触发器被触发。

  (4) 存储过程的调用(Stored procedure invocation)。

  为了响应数据库更新触,发器可以调用一个或多个存储过程,甚至可以通过外部过程的调用而在DBMS( 数据库管理系统)本身之外进行操作。

  由此可见,触发器可以解决高级形式的业务规则或复杂行为限制以及实现定制记录等一些方面的问题。例如,触发器能够找出某一表在数据修改前后状态发生的差异,并根据这种差异执行一定的处理。此外一个表的同一类型(Insert、 Update、 Delete)的多个触发器能够对同一种数据操作采取多种不同的处理。

  总体而言,触发器性能通常比较低。当运行触发器时,系统处理的大部分时间花费在参照其它表的这一处理上,因为这些表既不在内存中也不在数据库设备上,而删除表和插入表总是位于内存中。可见触发器所参照的其它表的位置决定了操作要花费的时间长短。

  2.触发器的种类

  SQL Server 2000 支持两种类型的触发器:AFTER 触发器和INSTEAD OF 触发器。其中AFTER 触发器即为SQL Server 2000 版本以前所介绍的触发器。该类型触发器要求只有执行某一操作(Insert Update Delete) 之后,触发器才被触发,且只能在表上定义。可以为针对表的同一操作定义多个触发器。对于AFTER 触发器,可以定义哪一个触发器被最先触发,哪一个被最后触发,通常使用系统过程sp_settriggerorder 来完成此任务。

  INSTEAD OF 触发器表示并不执行其所定义的操作(Insert、 Update、 Delete),而仅是执行触发器本身。既可在表上定义INSTEAD OF 触发器,也可以在视图上定义INSTEAD OF 触发器,但对同一操作只能定义一个INSTEAD OF 触发器。
?
————————————————————————————————————-

sql server存储过程语法
存储过程就是作为可执行对象存放在数据库中的一个或多个SQL命令。
??? 定义总是很抽象。存储过程其实就是能完成一定操作的一组SQL语句,只不过这组语句是放在数据库中的(这里我们只谈SQL Server)。如果我们通过创建存储过程以及在ASP中调用存储过程,就可以避免将SQL语句同ASP代码混杂在一起。这样做的好处至少有三个:
??? 第一、大大提高效率。存储过程本身的执行速度非常快,而且,调用存储过程可以大大减少同数据库的交互次数。
??? 第二、提高安全性。假如将SQL语句混合在ASP代码中,一旦代码失密,同时也就意味着库结构失密。
??? 第三、有利于SQL语句的重用。
??
??? 在ASP中,一般通过command对象调用存储过程,根据不同情况,本文也介绍其它调用方法。为了方便说明,根据存储过程的输入输出,作以下简单分类:
??? 1. 只返回单一记录集的存储过程
??? 假设有以下存储过程(本文的目的不在于讲述T-SQL语法,所以存储过程只给出代码,不作说明):

??? /*SP1*/
??? CREATE PROCEDURE dbo.getUserList
??? as
??? set nocount on
??? begin
?????? select * from dbo.[userinfo]
??? end
??? go

??? 以上存储过程取得userinfo表中的所有记录,返回一个记录集。通过command对象调用该存储过程的ASP代码如下:
??
??? ‘**通过Command对象调用存储过程**
??? DIM MyComm,MyRst
??? Set MyComm = Server.CreateObject(“ADODB.Command”)
??? MyComm.ActiveConnection = MyConStr????????? ‘MyConStr是数据库连接字串
??? MyComm.CommandText????? = “getUserList”???? ‘指定存储过程名
??? MyComm.CommandType????? = 4???????????????? ‘表明这是一个存储过程
??? MyComm.Prepared???????? = true????????????? ‘要求将SQL命令先行编译
??? Set MyRst = MyComm.Execute
??? Set MyComm = Nothing

??? 存储过程取得的记录集赋给MyRst,接下来,可以对MyRst进行操作。
??? 在以上代码中,CommandType属性表明请求的类型,取值及说明如下:
????? -1?? 表明CommandText参数的类型无法确定

???? 1??? 表明CommandText是一般的命令类型
????? 2??? 表明CommandText参数是一个存在的表名称
????? 4??? 表明CommandText参数是一个存储过程的名称
??
??? 还可以通过Connection对象或Recordset对象调用存储过程,方法分别如下:

??? ‘**通过Connection对象调用存储过程**
??? DIM MyConn,MyRst
??? Set MyConn = Server.CreateObject(“ADODB.Connection”)
??? MyConn.open MyConStr??????????????????????????? ‘MyConStr是数据库连接字串
??? Set MyRst? = MyConn.Execute(“getUserList”,0,4)? ‘最后一个参断含义同CommandType
??? Set MyConn = Nothing

??? ‘**通过Recordset对象调用存储过程**
??? DIM MyRst
??? Set MyRst = Server.CreateObject(“ADODB.Recordset”)
??? MyRst.open “getUserList”,MyConStr,0,1,4
??? ‘MyConStr是数据库连接字串,最后一个参断含义与CommandType相同

??
??? 2. 没有输入输出的存储过程
??? 请看以下存储过程:

??? /*SP2*/
??? CREATE PROCEDURE dbo.delUserAll
??? as
??? set nocount on
??? begin
?????? delete from dbo.[userinfo]
??? end
??? go

??? 该存储过程删去userinfo表中的所有记录,没有任何输入及输出,调用方法与上面讲过的基本相同,只是不用取得记录集:

??? ‘**通过Command对象调用存储过程**
??? DIM MyComm
??? Set MyComm = Server.CreateObject(“ADODB.Command”)
??? MyComm.ActiveConnection = MyConStr????????? ‘MyConStr是数据库连接字串
??? MyComm.CommandText????? = “delUserAll”????? ‘指定存储过程名
??? MyComm.CommandType????? = 4???????????????? ‘表明这是一个存储过程
??? MyComm.Prepared???????? = true????????????? ‘要求将SQL命令先行编译
??? MyComm.Execute????????????????????????????? ‘此处不必再取得记录集

??? Set MyComm = Nothing

??? 当然也可通过Connection对象或Recordset对象调用此类存储过程,不过建立Recordset对象是为了取得记录集,在没有返回记录集的情况下,还是利用Command对象吧。

??? 3. 有返回值的存储过程
??? 在进行类似SP2的操作时,应充分利用SQL Server强大的事务处理功能,以维护数据的一致性。并且,我们可能需要存储过程返回执行情况,为此,将SP2修改如下:

??? /*SP3*/
??? CREATE PROCEDURE dbo.delUserAll
??? as
??? set nocount on
??? begin
?????? BEGIN TRANSACTION
?????? delete from dbo.[userinfo]
?????? IF @@error=0
????????? begin
???????????? COMMIT TRANSACTION
???????????? return 1
????????? end
?????? ELSE
????????? begin
???????????? ROLLBACK TRANSACTION
???????????? return 0
????????? end???????
?????? return
??? end
??? go

??? 以上存储过程,在delete顺利执行时,返回1,否则返回0,并进行回滚操作。为了在ASP中取得返回值,需要利用Parameters集合来声明参数:

??? ‘**调用带有返回值的存储过程并取得返回值**
??? DIM MyComm,MyPara
??? Set MyComm = Server.CreateObject(“ADODB.Command”)
??? MyComm.ActiveConnection = MyConStr????????? ‘MyConStr是数据库连接字串
??? MyComm.CommandText????? = “delUserAll”????? ‘指定存储过程名
??? MyComm.CommandType????? = 4???????????????? ‘表明这是一个存储过程
??? MyComm.Prepared???????? = true????????????? ‘要求将SQL命令先行编译
??? ‘声明返回值
??? Set Mypara = MyComm.CreateParameter(“RETURN”,2,4)

??? MyComm.Parameters.Append MyPara
??? MyComm.Execute
??? ‘取得返回值
??? DIM retValue
??? retValue = MyComm(0)??? ‘或retValue = MyComm.Parameters(0)
??? Set MyComm = Nothing
??
??? 在MyComm.CreateParameter(“RETURN”,2,4)中,各参数的含义如下:
??? 第一个参数(“RETURE”)为参数名。参数名可以任意设定,但一般应与存储过程中声明的参数名相同。此处是返回值,我习惯上设为”RETURE”;
??? 第二个参数(2),表明该参数的数据类型,具体的类型代码请参阅ADO参考,以下给出常用的类型代码:
??? adBigInt: 20 ;
??? adBinary : 128 ;
??? adBoolean: 11 ;
??? adChar: 129 ;
??? adDBTimeStamp: 135 ;
??? adEmpty: 0 ;
??? adInteger: 3 ;
??? adSmallInt: 2 ;
??? adTinyInt: 16 ;
??? adVarChar: 200 ;
??? 对于返回值,只能取整形,且-1到-99为保留值;
??? 第三个参数(4),表明参数的性质,此处4表明这是一个返回值。此参数取值的说明如下:
??? 0 : 类型无法确定; 1: 输入参数;2: 输入参数;3:输入或输出参数;4: 返回值
??
??? 以上给出的ASP代码,应该说是完整的代码,也即最复杂的代码,其实

??? Set Mypara = MyComm.CreateParameter(“RETURN”,2,4)
??? MyComm.Parameters.Append MyPara
??????
??? 可以简化为

??? MyComm.Parameters.Append MyComm.CreateParameter(“RETURN”,2,4)

??? 甚至还可以继续简化,稍后会做说明。
??? 对于带参数的存储过程,只能使用Command对象调用(也有资料说可通过Connection对象或Recordset对象调用,但我没有试成过)。

??? 4. 有输入参数和输出参数的存储过程
??? 返回值其实是一种特殊的输出参数。在大多数情况下,我们用到的是同时有输入及输出参数的存储过程,比如我们想取得用户信息表中,某ID用户的用户名,这时候,有一个输入参数—-用户ID,和一个输出参数—-用户名。实现这一功能的存储过程如下:

??? /*SP4*/
??? CREATE PROCEDURE dbo.getUserName
?????? @UserID int,
?????? @UserName varchar(40) output
??? as
??? set nocount on
??? begin
?????? if @UserID is null return
?????? select @UserName=username
?????????? from dbo.[userinfo]

 

相关文章

发表评论 取消回复

电子邮件地址不会被公开。 必填项已用*标注