必威体育Betway必威体育官网
当前位置:首页 > IT技术

SQLSERVER存储过程语法详解

时间:2019-10-11 05:45:24来源:IT技术作者:seo实验室小编阅读:70次「手机版」
 

sql 存储过程

1

2

3

4

5

6

7

8

9

10

11

CREATE PROC [ EDURE ] procedure_name [ ; number ]

    [ { @parameter data_type }

        [ VARYING ] [ = default ] [ OUTPUT ]

    ] [ ,...n ]

[ WITH

    { RECOMPILE | ENCRYPTION | RECOMPILE , ENCRYPTION } ]

[ FOR REPLICATION ]

AS sql_statement [ ...n ]

@parameter 

过程中的参数。在 Create PROCEDURE 语句中可以声明一个或多个参数。用户必须在执行过程时提供每个所声明参数的值(除非定义了该参数的默认值)。

OUTPUT 

表明参数是返回参数。该选项的值可以返回给 EXEC[UTE]。使用 OUTPUT 参数可将信息返回给调用过程。Text、ntext 和 image 参数可用作 OUTPUT 参数。使用 OUTPUT 关键字的输出参数可以是游标占位符。 

AS :指定过程要执行的操作

SQLSERVER:

变量的声明:

声明变量时必须在变量前加@符号 

DECLARE @I INT

变量的赋值:

变量赋值时变量前必须加set

SET @I = 30

声明多个变量:

DECLARE @s varchar(10),@a INT

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

oracle的建表sql转成sqlserver的建表sql时的注意点 :

1.所有的comment语句需要删除。

2.clob类型转换为text类型。

3.blob类型转换为image类型。

4.number类型转换为int,number(16,2)等转换为decimal(16,2),number(18)转换为bigint。

5.default sysdate改为default getDate()。

6.to_date('2009-12-18','yyyy-mm-dd')改为cast('2009-12-18'  as   datetime)

SQLSERVER:

变量的声明:

声明变量时必须在变量前加@符号

DECLARE @I INT

变量的赋值:

变量赋值时变量前必须加set

SET @I = 30

声明多个变量:

DECLARE @s varchar(10),@a INT

if语句:

java代码  收藏代码

  1. if ..  
  2. begin  
  3.   ...  
  4. end  
  5. else if ..  
  6. begin  
  7.   ...  
  8. end  
  9. else  
  10. begin  
  11.   ...  
  12. end   

example:

Sql代码  收藏代码

  1. DECLARE @d INT  
  2. set @d = 1  
  3. IF @d = 1 BEGIN  
  4.    print '正确'  
  5. END  
  6. ELSE BEGIN  
  7.    PRINT '错误'  
  8. END  

多条件选择语句:

Example:

Sql代码  收藏代码

  1. declare @today int  
  2. declare @week nvarchar(3)  
  3. set @today=3  
  4. set @week= case  
  5.      when @today=1 then '星期一'  
  6.      when @today=2 then '星期二'  
  7.      when @today=3 then '星期三'  
  8.      when @today=4 then '星期四'  
  9.      when @today=5 then '星期五'  
  10.      when @today=6 then '星期六'  
  11.      when @today=7 then '星期日'  
  12.      else '值错误'  
  13. end  
  14. print @week  

循环语句:

java代码  收藏代码

  1. while 条件 BEGIN    
  2. 执行语句  
  3. END    

Example:

Java代码  收藏代码

  1. DECLARE @i INT  
  2. SET @i = 1  
  3. WHILE @i<1000000 BEGIN  
  4. set @i=@i+1  
  5. END  

定义游标:

Sql代码  收藏代码

  1. DECLARE @cur1 CURSOR FOR SELECT .........  
  2.   
  3. OPEN @cur1  
  4. FETCH NEXT FROM @cur1 INTO 变量  
  5. WHILE(@@FETCH_STATUS=0)  
  6. BEGIN  
  7. 处理.....  
  8. FETCH NEXT FROM @cur1 INTO 变量  
  9. END  
  10. CLOSE @cur1  
  11. deallocate @cur1   

Sql代码  收藏代码

  1. AS  
  2.   
  3. declare @CATEGORY_CI_TABLENAME VARCHAR(50) =''  
  4. declare @result VARCHAR(2000) = ''  
  5. declare @CI_ID DECIMAL = 0  
  6. declare @num int = 1  
  7. declare @countnum int = 1  
  8.   
  9. BEGIN  
  10. select  @countnum = count(ATTRIBUTE_CONFIG_ID) from T_ATTRIBUTE_CONFIG where CMDB_UPDATE_FLAG= 'Y' and CATEGORY_CODE =@CATEGORY_CODE  
  11.    
  12. IF (@ATTRIBUTE2='A')  
  13.   begin    
  14.         DECLARE MyCursor CURSOR for select ATTRIBUTE_CONFIG_CODE from T_ATTRIBUTE_CONFIG where  CMDB_UPDATE_FLAG= 'Y' and CATEGORY_CODE =@CATEGORY_CODE  
  15.          OPEN MyCursor FETCH NEXT FROM MyCursor INTO @CONFIG_CODE  
  16.                 set @result = @result+@CONFIG_CODE+','  
  17.              WHILE @@FETCH_STATUS = 0  
  18.                     BEGIN  
  19.                     FETCH NEXT FROM MyCursor INTO @CONFIG_CODE  
  20.                     set @num = @num+ 1  
  21.                         if(@num<@countnum)   
  22.                             begin  
  23.                                 set @result = @result+@CONFIG_CODE+','  
  24.                             end   
  25.                         else if(@num=@countnum)   
  26.                              begin  
  27.                                 set @result = @result +@CONFIG_CODE  
  28.                              end   
  29.                     END  
  30.             CLOSE MyCursor   
  31.             DEALLOCATE MyCursor   
  32.         set @result = 'insert into ' + @ATTRIBUTE1 + '(' + @result +') select '+ @result +' from '+@CATEGORY_CI_TABLENAME +' where CI_order_LINE_ID='+@KEY_ID  
  33.   end             
  34.  else if((@ATTRIBUTE2='U'))  

临时表:

-- Select INTO 从一个查询的计算结果中创建一个新表数据并不返回给客户端,这一点和普通的Select 不同。 新表的字段具有和 Select 的输出字段相关联(相同)的名字和数据类型。

   select * into NewTable

   from Uname

-- Insert INTO ABC Select

   -- 表ABC必须存在 

   -- 把表Uname里面的字段Username复制到表ABC

   Insert INTO ABC Select Username FROM Uname

-- 创建临时表

   Create TABLE #temp(

   UID int identity(1, 1) PRIMARY KEY,

   UserName varchar(16),

   Pwd varchar(50),

   Age smallint,

   Sex varchar(6)

   )

       

-- 打开临时表

   Select * from #temp

1、局部临时表(#开头)只对当前连接有效,当前连接断开时自动删除。

2、全局临时表(##开头)对其它连接也有效,在当前连接和其他访问过它的连接都断开时自动删除。

3、不管局部临时表还是全局临时表,只要连接有访问权限,都可以用drop table #Tmp(或者drop table ##Tmp)来显式删除临时表。

临时表对执行效率应该影响不大,只要不是太过份,相反可以提高效率特别是连接查询的地方,只要你的数据库临时表空间足够

游标多,会严重执行效率,能免则免!

临时表在不同数据库设计中的作用

SQLSERVER 存储过程 语法

===============================================================================

其他:

--有输入参数的存储过程--

create proc GetComment

(@commentid int)

as

select * from Comment where CommentID=@commentid

--有输入与输出参数的存储过程--

create proc GetCommentCount

@newsid int,

@count int output

as

select @count=count(*) from Comment where NewsID=@newsid

--返回单个值的函数--

create function MyFunction

(@newsid int)

returns int

as

begin

declare @count int

select @count=count(*) from Comment where NewsID=@newsid

return @count

end

--调用方法--

declare @count int

exec @count=MyFunction 2

print @count

--返回值为表的函数--

Create function GetFunctionTable

(@newsid int)

returns table

as

return

(select * from Comment where NewsID=@newsid)

--返回值为表的函数的调用--

select * from GetFunctionTable(2)

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

SQLServer 存储过程中不拼接SQL字符串实现多条件查询

 以前拼接的写法

set @sql=' select * from table where 1=1 '

if (@addDate is not null) 

set @sql = @sql+' and addDate = '+ @addDate + ' ' 

if (@name <>'' and is not null) 

set @sql = @sql+ ' and name = ' + @name + ' '

exec(@sql)

下面是 不采用拼接SQL字符串实现多条件查询的解决方案

第一种写法是 感觉代码有些冗余

if (@addDate is not null) and (@name <> '') 

select * from table where addDate = @addDate and name = @name 

else if (@addDate is not null) and (@name ='') 

select * from table where addDate = @addDate 

else if(@addDate is null) and (@name <> '') 

select * from table where and name = @name 

else if(@addDate is null) and (@name = '') 

select * from table 

第二种写法是 

select * from table where (addDate = @addDate or @addDate is null) and (name = @name or @name = '') 

第三种写法是 

SELECT * FROM table where 

addDate = CASE @addDate IS NULL THEN addDate ELSE @addDate END, 

name = CASE @name WHEN '' THEN name ELSE @name END

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

SQLSERVER存储过程基本语法

一、定义变量

--简单赋值
declare @a  int
set @a=5
print @a
   
--使用select语句赋值
declare @user1 nvarchar(50)
select @user1= '张三'
print @user1
declare @user2 nvarchar(50)
select @user2 =  Name from ST_User  where ID=1
print @user2
   
--使用update语句赋值
declare @user3 nvarchar(50)
update ST_User  set @user3 =  Name where ID=1
print @user3

二、表、临时表、表变量

--创建临时表1
create table #DU_User1
(
      [ID] [ int  NOT NULL ,
      [Oid] [ int ]  NOT NULL ,
      [Login] [nvarchar](50)  NOT NULL ,
      [Rtx] [nvarchar](4)  NOT NULL ,
      [ Name ] [nvarchar](5)  NOT NULL ,
      [ Password ] [nvarchar]( max )  NULL ,
      [State] [nvarchar](8)  NOT NULL
);
--向临时表1插入一条记录
insert into #DU_User1 (ID,Oid,[Login],Rtx, Name ,[ Password ],State)  values (100,2, 'LS' ,'0000' , '临时' , '321' , '特殊' );
   
--从ST_User查询数据,填充至新生成的临时表
select *  into #DU_User2  from ST_User  where ID<8
   
--查询并联合两临时表
select *  from #DU_User2  where ID<3  union select *  from #DU_User1
   
--删除两临时表
drop table #DU_User1
drop table #DU_User2
--创建临时表
CREATE TABLE #t
(
     [ID] [ int ]  NOT NULL ,
     [Oid] [ int ]  NOT NULL ,
     [Login] [nvarchar](50)  NOT NULL ,
     [Rtx] [nvarchar](4)  NOT NULL ,
     [ Name ] [nvarchar](5)  NOT NULL ,
     [ Password ] [nvarchar]( max )  NULL ,
     [State] [nvarchar](8)  NOT NULL ,
)
   
--将查询结果集(多条数据)插入临时表
insert into #t  select *  from ST_User
--不能这样插入
--select * into #t from dbo.ST_User
   
--添加一列,为int型自增长子段
alter table #t  add [myid]  int NOT NULL IDENTITY(1,1)
--添加一列,默认填充全球唯一标识
alter table #t  add [myid1] uniqueidentifier  NOT NULL default (newid())
   
select *  from #t
drop table #t
--给查询结果集增加自增长列
   
--无主键时:
select IDENTITY( int ,1,1) as ID,  Name ,[Login],[ Password ]  into #t  from ST_User
select *  from #t
   
--有主键时:
select ( select SUM (1)  from ST_User  where ID<= a.ID)  as myID,*  from ST_User a  order bymyID
--定义表变量
declare @t  table
(
     id  int not null ,
     msg nvarchar(50)  null
)
insert into @t  values (1, '1' )
insert into @t  values (2, '2' )
select *  from @t

三、循环

--while循环计算1到100的和
declare @a  int
declare @ sum int
set @a=1
set @ sum =0
while @a<=100
begin
     set @ sum +=@a
     set @a+=1
end
print @ sum

四、条件语句

--if,else条件分支
if(1+1=2)
begin
     print  '对'
end
else
begin
     print  '错'
end
   
--when then条件分支
declare @today  int
declare @week nvarchar(3)
set @today=3
set @week= case
     when @today=1  then '星期一'
     when @today=2  then '星期二'
     when @today=3  then '星期三'
     when @today=4  then '星期四'
     when @today=5  then '星期五'
     when @today=6  then '星期六'
     when @today=7  then '星期日'
     else '值错误'
end
print @week

五、游标

declare @ID  int
declare @Oid  int
declare @Login  varchar (50)
   
--定义一个游标
declare user_cur  cursor for select ID,Oid,[Login]  from ST_User
--打开游标
open user_cur
while @@fetch_status=0
begin
--读取游标
     fetch next from user_cur  into @ID,@Oid,@Login
     print @ID
     --print @Login
end
close user_cur
--摧毁游标
deallocate user_cur

六、触发器

   触发器中的临时表:

Inserted 

存放进行insert和update 操作后的数据 

Deleted 

存放进行delete 和update操作前的数据

--创建触发器
Create trigger User_OnUpdate 
     On ST_User 
     for Update  
As  
     declare @msg nvarchar(50)
     --@msg记录修改情况
     select @msg = N '姓名从“' + Deleted. Name + N '”修改为“' + Inserted. Name +  '”' fromInserted,Deleted
     --插入日志表
     insert into [LOG](MSG) values (@msg)
       
--删除触发器
drop trigger User_OnUpdate

七、存储过程

--创建带output参数的存储过程
CREATE PROCEDURE PR_Sum
     @a  int ,
     @b  int ,
     @ sum int output
AS
BEGIN
     set @ sum =@a+@b
END
   
--创建Return返回值存储过程
CREATE PROCEDURE PR_Sum2
     @a  int ,
     @b  int
AS
BEGIN
     Return @a+@b
END
       
--执行存储过程获取output型返回值
declare @mysum  int
execute PR_Sum 1,2,@mysum  output
print @mysum
   
--执行存储过程获取Return型返回值
declare @mysum2  int
execute @mysum2= PR_Sum2 1,2
print @mysum2

  

八、自定义函数

函数的分类:

  1)标量值函数

  2)表值函数

      a:内联表值函数

      b:多语句表值函数

  3)系统函数

--新建标量值函数
create function FUNC_Sum1
(
     @a  int ,
     @b  int
)
returns int
as
begin
     return @a+@b
end
   
--新建内联表值函数
create function FUNC_UserTab_1
(
     @myId  int
)
returns table
as
return ( select *  from ST_User  where ID<@myId)
   
--新建多语句表值函数
create function FUNC_UserTab_2
(
     @myId  int
)
returns @t  table
(
     [ID] [ int ]  NOT NULL ,
     [Oid] [ int ]  NOT NULL ,
     [Login] [nvarchar](50)  NOT NULL ,
     [Rtx] [nvarchar](4)  NOT NULL ,
     [ Name ] [nvarchar](5)  NOT NULL ,
     [ Password ] [nvarchar]( max )  NULL ,
     [State] [nvarchar](8)  NOT NULL
)
as
begin
     insert into @t  select *  from ST_User  where ID<@myId
     return
end
   
--调用表值函数
select *  from dbo.FUNC_UserTab_1(15)
--调用标量值函数
declare @s  int
set @s=dbo.FUNC_Sum1(100,50)
print @s
   
--删除标量值函数
drop function FUNC_Sum1

谈谈自定义函数与存储过程的区别:

一、自定义函数:

1. 可以返回表变量

2. 限制颇多,包括

  不能使用output参数;

  不能用临时表;

  函数内部的操作不能影响到外部环境;

  不能通过select返回结果集;

  不能update,delete,数据库表;

3. 必须return 一个标量值或表变量

自定义函数一般用在复用度高,功能简单单一,争对性强的地方。

二、存储过程

1. 不能返回表变量

2. 限制少,可以执行对数据库表的操作,可以返回数据集

3. 可以return一个标量值,也可以省略return

 存储过程一般用在实现复杂的功能,数据操纵方面。

 

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

SqlServer存储过程--实例

实例1:只返回单一记录集的存储过程。

表银行存款表(bankMoney)的内容如下

Id

userID

Sex

Money

001

Zhangsan

30

002

Wangwu

50

003

Zhangsan

40

要求1:查询表bankMoney的内容的存储过程

create procedure sp_query_bankMoney

as

select * from bankMoney

go

exec sp_query_bankMoney

注*  在使用过程中只需要把T-Sql中的SQL语句替换为存储过程名,就可以了很方便吧!

实例2(向存储过程中传递参数):

加入一笔记录到表bankMoney,并查询此表中userID= Zhangsan的所有存款的总金额。

Create proc insert_bank @param1 char(10),@param2 varchar(20),@param3 varchar(20),@param4 int,@param5 int output

with encryption ---------加密

as

insert into bankMoney (id,userID,sex,Money)

Values(@param1,@param2,@param3, @param4)

select @param5=sum(Money) from bankMoney where userID='Zhangsan'

go

在SQL Server查询分析器中执行该存储过程的方法是:

declare @total_price int

exec insert_bank '004','Zhangsan','男',100,@total_price output

print '总余额为'+convert(varchar,@total_price)

go

在这里再啰嗦一下存储过程的3种传回值(方便正在看这个例子的朋友不用再去查看语法内容):

1.以Return传回整数

2.以output格式传回参数

3.Recordset

传回值的区别:

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

实例3:使用带有复杂 SELECT 语句的简单过程

下面的存储过程从四个表的联接中返回所有作者(提供了姓名)、出版的书籍以及出版社。该存储过程不使用任何参数。

USE pubs

IF exists (SELECT name FROM sysobjects

WHERE name = 'au_info_all' AND type = 'P')

  DROP PROCEDURE au_info_all

GO

CREATE PROCEDURE au_info_all

AS

SELECT au_lname, au_fname, title, pub_name

  FROM authors a inner join titleauthor ta

ON a.au_id = ta.au_id INNER JOIN titles t

ON t.title_id = ta.title_id INNER JOIN publishers p

ON t.pub_id = p.pub_id

GO

au_info_all 存储过程可以通过以下方法执行:

EXECUTE au_info_all

-- Or

EXEC au_info_all

如果该过程是批处理中的第一条语句,则可使用:

au_info_all

实例4:使用带有参数的简单过程

CREATE PROCEDURE au_info

  @lastname varchar(40),

  @firstname varchar(20)

AS

SELECT au_lname, au_fname, title, pub_name

  FROM authors a INNER JOIN titleauthor ta

ON a.au_id = ta.au_id INNER JOIN titles t

ON t.title_id = ta.title_id INNER JOIN publishers p

ON t.pub_id = p.pub_id

  WHERE  au_fname = @firstname

AND au_lname = @lastname

GO

au_info 存储过程可以通过以下方法执行:

EXECUTE au_info 'Dull', 'Ann'

-- Or

EXECUTE au_info @lastname = 'Dull', @firstname = 'Ann'

-- Or

EXECUTE au_info @firstname = 'Ann', @lastname = 'Dull'

-- Or

EXEC au_info 'Dull', 'Ann'

-- Or

EXEC au_info @lastname = 'Dull', @firstname = 'Ann'

-- Or

EXEC au_info @firstname = 'Ann', @lastname = 'Dull'

如果该过程是批处理中的第一条语句,则可使用:

au_info 'Dull', 'Ann'

-- Or

au_info @lastname = 'Dull', @firstname = 'Ann'

-- Or

au_info @firstname = 'Ann', @lastname = 'Dull'

实例5:使用带有通配符参数的简单过程

CREATE PROCEDURE au_info2

@lastname varchar(30) = 'D%',

@firstname varchar(18) = '%'

AS

SELECT au_lname, au_fname, title, pub_name

FROM authors a INNER JOIN titleauthor ta

   ON a.au_id = ta.au_id INNER JOIN titles t

   ON t.title_id = ta.title_id INNER JOIN publishers p

   ON t.pub_id = p.pub_id

WHERE au_fname LIKE @firstname

   AND au_lname LIKE @lastname

GO

au_info2 存储过程可以用多种组合执行。下面只列出了部分组合:

EXECUTE au_info2

-- Or

EXECUTE au_info2 'Wh%'

-- Or

EXECUTE au_info2 @firstname = 'A%'

-- Or

EXECUTE au_info2 '[CK]ars[OE]n'

-- Or

EXECUTE au_info2 'Hunter', 'Sheryl'

-- Or

EXECUTE au_info2 'H%', 'S%'

= 'proc2'

实例6:if...else

存储过程,其中@case作为执行update的选择依据,用if...else实现执行时根据传入的参数执行不同的修改. 

--下面是if……else的存储过程: 

if exists (select 1 from sysobjects where name = 'Student' and type ='u' )

drop table Student

go

if exists (select 1 from sysobjects where name = 'spUpdateStudent' and type ='p' )

drop proc spUpdateStudent

go

create table Student

(

fName nvarchar (10),

fAge 

smallint ,

fDiqu varchar (50),

fTel  int 

)

go

insert into Student values ('X.X.Y' , 28, 'Tesing' , 888888)

go

create proc spUpdateStudent

(

@fCase int ,

@fName nvarchar (10),

@fAge smallint ,

@fDiqu varchar (50),

@fTel  int 

)

as 

update Student

set fAge = @fAge, -- 传 1,2,3 都要更新 fAge 不需要用 case 

fDiqu = (case when @fCase = 2 or @fCase = 3 then @fDiqu else fDiqu end ),

fTel  = (case when @fCase = 3 then @fTel else fTel end )

where fName = @fName

select * from Student

go

-- 只改 Age 

exec spUpdateStudent

@fCase = 1,

@fName = N'X.X.Y' ,

@fAge = 80,

@fDiqu = N'Update' ,

@fTel  = 1010101

-- 改 Age 和 Diqu 

exec spUpdateStudent

@fCase = 2,

@fName = N'X.X.Y' ,

@fAge = 80,

@fDiqu = N'Update' ,

@fTel  = 1010101

-- 全改 

exec spUpdateStudent

@fCase = 3,

@fName = N'X.X.Y' ,

@fAge = 80,

@fDiqu = N'Update' ,

@fTel  = 1010101

相关阅读

从浏览器输入一个URL(www.baidu.com)的全过程

1.根据域名到DNS中找到IP 2.根据IP建立TCP连接(三次握手) 3.连接建立成功发起http请求 4.服务器响应http请求 5.浏览器解析HTML代

SqlServer之like、charindex、patindex区别及性能分析

第一篇文章:SqlServer之like、charindex、patindex1、环境介绍测试环境 SQL2005测试数据 200W条2、环境准备2.1建表CREATE TABLE [

产品经理日报第1030期|腾讯副总裁丁珂:微信不会读取、存

哈喽,你我相约七点半,你来了么^_^产品经理日报继续为您带来今日最新的资讯:美团正式入局共享充电宝,行业竞争加剧;Snapchat新功能,用人

Ceph存储使用RBD(Rados块设备)

使用RBD(Rados块设备)查看存储池[root@node1~]# ceph osd lspools可以查看到0号镜像池,名字为rbd创建名为demo-img的镜像大小为1

win7 系统装SQLServer2000 成功。

原文地址为:win7 系统装SQLServer2000 成功。昨天在win7上装SQLServer数据库,写一下体会。首先,如果以前安装的话,要删除干净。我也找

分享到:

栏目导航

推荐阅读

热门阅读