流程控制语句

顺序语句

①set语句
  set语句有两种用法,除了用于给局部变量赋值,还可以设定用户执行T-SQL命令时SQL Server的处理选项,一般有以下几种设定方式。

  • set 选项 on:选项开关打开。
  • set 选项 off:选项开关关闭。
  • set 选项值:设定选项的具体值

例如,设置显示/隐藏受T-SQL语句影响的行数消息语句,其语法如下:

set nocount(on|off)

②select输出语句和print输出语句
  SQL Server使用print或select输出信息。
  select作为输出使用时的语法如下:

select 表达式1[,表达式2,...,表达式n]

print用于向屏幕输出信息,其语法如下:

print 表达式
  • select输出会包含分隔线
  • print输出类型一定要是字符类型

【示例】

1> print 'Hello World!'
2> go
Hello World!
1> select 'Hello World!'
2> go

------------
Hello World!

分支语句

①IF...ELSE语句

IF Boolean_expression   
     { sql_statement | statement_block }
[ELSE   
     { sql_statement | statement_block } ]   

说明:

  • IF...ELSE语句间不只包含一条语句时,必须使用BEGIN...END包含多条语句的语句块。
  • 语句块的关键字BEGIN、END必须成对出现,允许嵌套,并且BEGIN...END之间至少包含一条SQL语句。
  • IF...ELSE可以嵌套使用

【示例】

DECLARE @Number int;  
SET @Number = 50;  
IF @Number > 100  
   PRINT 'The number is large.';  
ELSE   
   BEGIN  
      IF @Number < 10  
      PRINT 'The number is small.';  
   ELSE  
      PRINT 'The number is medium.';  
   END ;  
GO  

②CASE函数
  CASE是计算条件列表,并返回多个可能的结果表达式之一。
  CASE 表达式有两种格式:

  • CASE 简单表达式,它通过将表达式与一组简单的表达式进行比较来确定结果。
  • CASE 搜索表达式,它通过计算一组布尔表达式来确定结果。

这两种格式都支持可选的 ELSE 参数。
  CASE 可用于允许使用有效表达式的任意语句或子句。 例如,可以在 SELECT、UPDATE、DELETE 和 SET 等语句以及 select_list、IN、WHERE、ORDER BY 和 HAVING 等子句中使用 CASE。
  CASE语法:

--简单Case函数
CASE input_expression   
     WHEN when_expression THEN result_expression [ ...n ]   
     [ ELSE else_result_expression ]   
END   
--Case搜索函数
CASE  
     WHEN Boolean_expression THEN result_expression [ ...n ]   
     [ ELSE else_result_expression ]   
END  

【返回类型】从的类型集中返回优先级高的类型result_expressions和可选else_result_expression。

CASE 表达式不能用于控制 Transact-SQL 语句、语句块、用户定义函数以及存储过程的执行流。能控制的只有IF..ELSE、WHILE这样的控制流语言。

【示例】
A. 使用带有 CASE 简单表达式的 SELECT 语句

SELECT   ProductNumber, Category =  
      CASE ProductLine  
         WHEN 'R' THEN 'Road'  
         WHEN 'M' THEN 'Mountain'  
         WHEN 'T' THEN 'Touring'  
         WHEN 'S' THEN 'Other sale items'  
         ELSE 'Not for sale'  
      END,  
   Name  
FROM Production.Product  
ORDER BY ProductNumber;  
GO  

B. 使用带有 CASE 搜索表达式的 SELECT 语句
在 SELECT 语句中,CASE 搜索表达式允许根据比较值替换结果集中的值。 下面的示例根据产品的价格范围将标价显示为文本注释。

SELECT   ProductNumber, Name, "Price Range" =   
      CASE   
         WHEN ListPrice =  0 THEN 'Mfg item - not for resale'  
         WHEN ListPrice < 50 THEN 'Under $50'  
         WHEN ListPrice >= 50 and ListPrice < 250 THEN 'Under $250'  
         WHEN ListPrice >= 250 and ListPrice < 1000 THEN 'Under $1000'  
         ELSE 'Over $1000'  
      END  
FROM Production.Product  
ORDER BY ProductNumber ;  
GO  

C. 在 ORDER BY 子句中使用 CASE
下面的示例在 ORDER BY 子句中使用 CASE 表达式,以根据给定的列值确定行的排序顺序。

SELECT BusinessEntityID, SalariedFlag  
FROM HumanResources.Employee  
ORDER BY CASE SalariedFlag WHEN 1 THEN BusinessEntityID END DESC  
        ,CASE WHEN SalariedFlag = 0 THEN BusinessEntityID END;  
GO  

D. 在 SET 语句中使用 CASE

SET @ContactType =   
        CASE   
            -- Check for employee  
            WHEN EXISTS(SELECT * FROM HumanResources.Employee AS e   
                WHERE e.BusinessEntityID = @BusinessEntityID)   
                THEN 'Employee'  
  
            -- Check for vendor  
            WHEN EXISTS(SELECT * FROM Person.BusinessEntityContact AS bec  
                WHERE bec.BusinessEntityID = @BusinessEntityID)   
                THEN 'Vendor'  
  
            -- Check for store  
            WHEN EXISTS(SELECT * FROM Purchasing.Vendor AS v            
                WHERE v.BusinessEntityID = @BusinessEntityID)   
                THEN 'Store Contact'  
  
            -- Check for individual consumer  
            WHEN EXISTS(SELECT * FROM Sales.Customer AS c   
                WHERE c.PersonID = @BusinessEntityID)   
                THEN 'Consumer'  
        END;  

循环语句

T-SQL中的循环语句只有while一种语法形式,while语句还可以和break、continue语句一起使用。其结构如下所示:

WHILE Boolean_expression   
     { sql_statement | statement_block | BREAK | CONTINUE }  
  • WHILE语句可以嵌套
  • WHILE语句间不只包含一条语句时,必须使用BEGIN...END包含多条语句的语句块。
  • CONTINUE:使 WHILE 循环重新开始执行,忽略 CONTINUE 关键字后面的任何语句。
  • BREAK:导致从最内层的 WHILE 循环中退出。 执行标记的末尾循环中,在结束关键字后显示的任何语句。
  • 如果嵌套了两个或多个 WHILE 循环,则内层的 BREAK 将退出到下一个外层循环。 将首先运行内层循环结束之后的所有语句,然后重新开始下一个外层循环。

【示例】

USE AdventureWorks2012;  
GO  
WHILE (SELECT AVG(ListPrice) FROM Production.Product) < $300  
BEGIN  
   UPDATE Production.Product SET ListPrice = ListPrice * 2  
   SELECT MAX(ListPrice) FROM Production.Product  
   IF (SELECT MAX(ListPrice) FROM Production.Product) > $500  
      BREAK  
   ELSE  
      CONTINUE  
END  
PRINT 'Too much for the market to bear';  

其他控制语句

①WAITFOR语句
  waitfor语句称为延时语句,它的功能暂停程序执行,直到所设定的等待时间已过或所设定的时间已到才继续玩下执行。。其语法如下:

WAITFOR   
{  
    DELAY 'time_to_pass'   
  | TIME 'time_to_execute'   
  | [ ( receive_statement ) | ( get_conversation_group_statement ) ]   
    [ , TIMEOUT timeout ]  
}  
  • DELAY:用来设定等待时间,最多可达24小时。
  • time_to_pass:等待的时段。 time_to_pass可接受格式之一指定datetime数据,也可以指定为本地变量。
  • TIME:指定的运行批处理、存储过程或事务的时间。
  • time_to_execute:WAITFOR 语句完成的时间。 time_to_execute可接受格式之一指定datetime数据,也可以指定为本地变量。

【示例】
A. 使用 WAITFOR TIME
下面的示例在晚上 10:20 在 msdb 数据库中执行 sp_update_job 存储过程。

EXECUTE sp_add_job @job_name = 'TestJob';  
BEGIN  
    WAITFOR TIME '22:20';  
    EXECUTE sp_update_job @job_name = 'TestJob',  
        @new_name = 'UpdatedJob';  
END;  
GO  

B. 使用 WAITFOR DELAY
以下示例在两小时的延迟后执行存储过程。

BEGIN  
    WAITFOR DELAY '02:00';  
    EXECUTE sp_helpdb;  
END;  
GO  

C. 在 WAITFOR DELAY 中使用局部变量
以下示例显示如何对 WAITFOR DELAY 选项使用局部变量。 将创建一个存储过程,该过程将等待可变的时间段,然后将经过的小时、分钟和秒数信息返回给用户。

CREATE PROCEDURE dbo.TimeDelay_hh_mm_ss   
(@DelayLength char(8)= '00:00:00')  
AS  
DECLARE @ReturnInfo varchar(255)  
IF ISDATE('2000-01-01 ' + @DelayLength + '.000') = 0  
    BEGIN  
        SELECT @ReturnInfo = 'Invalid time ' + @DelayLength   
        + ',hh:mm:ss, submitted.';
        PRINT @ReturnInfo   
        RETURN(1)  
    END  
BEGIN  
    WAITFOR DELAY @DelayLength  
    SELECT @ReturnInfo = 'A total time of ' + @DelayLength + ',   
        hh:mm:ss, has elapsed! Your time is up.'  
    PRINT @ReturnInfo;  
END;  
GO  
EXEC TimeDelay_hh_mm_ss '00:00:10';  
GO  

②RETURN语句
  从查询或过程中无条件退出。 RETURN 的执行是即时且完全的,可在任何时候用于从过程、批处理或语句块中退出。 RETURN 之后的语句是不执行的。其语法如下:

RETURN [<整形表达式>]

错误处理语句

a. 简单的异常处理
  在T-SQL程序中,把可能出现异常的程序语句放置在try部分(begin try...end tey),错误处理语句放置在catch部分(begin catch...end catch),如果在try部分未发生错误,则catch部分的语句会被忽略,如果try部分的语句发生了异常,则程序控制转移至相应的catch部分。如果try..catch捕获了错误并进行了处理,对于调用者来说,相当于没发生错误。
  如下面语句在try部分不会发生错误,catch部分的语句也不会被执行:

begin try
  print 10/2;
  print 'No error';
end try
begin catch
  print 'Error!Divided by zero.'
end catch

b. 预定义异常
  在一般T-SQL程序设计中,编程者应该在catch部分分析错误原因,然后给出相应提示信息或其他处理语句。
  catch部分捕获到的错误以错误号来标识,编程者通过调用error-numer()函数得到捕获到的错误号,然后根据错误号给出相关错误提示信息,或者调用error-message()函数显示系统预先定义的错误信息,SQL Server的所有预定以错误号及对应错误信息可以通过查询sys.message得到。另外,SQL Server不支持用户自定义异常。
  下面示例说明try...catch的用法。创建表t,用于测试:

1> create table t
2> (
3>   a int not null,
4>   b int not null,
5>   c char(4),
6>   constraint pk_t primary key(a),
7>   constraint ck_b check(b>100)
8> )

然后执行下面程序段:

begin try
  insert into t values(1,170,'a');
end try
begin catch
  if error_number() = 2627
  begin 
    print 'Handling PK violation...';
  end

  else if error_number() = 547
  begin 
    print 'Handling CHECK/FK constraint violation...';
  end

  else if error_number() = 515
  begin 
    print 'Handling NULL violation...';
  end

  else if error_number() = 245
  begin 
    print 'Handling conversion error...';
  end

  else
  begin 
    print 'Handling unknow error...';
  end

-- 输出错误号
print 'Error Number:'+cast(error_number() as varchar(10));
-- 输出错误信息
print 'Error Message:'+error_message();
print 'Error Severity:'+cast(error_severity() as varchar(10));
print 'Error State:'+cast(error_state() as varchar(10));
print 'Error Line:'+cast(error_line() as varchar(10));
print 'Error Proc:'+coalesce(error_procedure(),'Not within proc');

end catch

如果第二次执行上面的语句,则会因为主键存在重复值而给出以下错误信息:

Handling PK violation...
Error Number:2627
Error Message:违反了 PRIMARY KEY 约束 'pk_t'。不能在对象 'dbo.t' 中插入重复键。
Error Severity:14
Error State:1
Error Line:2
Error Proc:Not within proc

如果把添加记录的insert语句改为:
  insert into t values(2,10,'a');
  修改后,重新执行上述程序段,则违反了b字段上的check约束,会给出以下错误信息:

Handling CHECK/FK constraint violation...
Error Number:547
Error Message:INSERT 语句与 CHECK 约束"ck_b"冲突。该冲突发生于数据库"master", 表"dbo.t", column 'b'。
Error Severity:16
Error State:0
Error Line:2
Error Proc:Not within proc

c. 使用存储过程处理异常
  如果要使得错误处理的代码可以重用,则可以把上述catch的内容创建为存储过程,以后即可在T-SQL代码中直接调用。
  创建存储过程如下:

create procedure dbo.err_message
as
 if error_number() = 2627
  begin 
    print 'Handling PK violation...';
  end

  else if error_number() = 547
  begin 
    print 'Handling CHECK/FK constraint violation...';
  end

  else if error_number() = 515
  begin 
    print 'Handling NULL violation...';
  end

  else if error_number() = 245
  begin 
    print 'Handling conversion error...';
  end

  else
  begin 
    print 'Handling unknow error...';
  end

-- 输出错误号
print 'Error Number:'+cast(error_number() as varchar(10));
-- 输出错误信息
print 'Error Message:'+error_message();
print 'Error Severity:'+cast(error_severity() as varchar(10));
print 'Error State:'+cast(error_state() as varchar(10));
print 'Error Line:'+cast(error_line() as varchar(10));
print 'Error Proc:'+coalesce(error_procedure(),'Not within proc');

在T-SQL程序段中调用上述存储过程:

begin try
  insert into t values(2,10,'a');
end try
begin catch
  exec err_message;
end catch
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 215,294评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,780评论 3 391
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 161,001评论 0 351
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,593评论 1 289
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,687评论 6 388
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,679评论 1 294
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,667评论 3 415
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,426评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,872评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,180评论 2 331
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,346评论 1 345
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 35,019评论 5 340
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,658评论 3 323
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,268评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,495评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,275评论 2 368
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,207评论 2 352

推荐阅读更多精彩内容