在SQL Server2005中进行错误捕捉

任何程序都可能出现错误,在SQL Server中执行Transact-SQL也不例外。如果在Transact-SQL中发生了错误,一般有两种捕捉错误的方法,一种是在客户端代码(如 c#、delphi等)中使用类似try...catch的语句进行捕捉;另外一种就是在Transact-SQL中利用Transact-SQL本身提供的错误捕捉机制进行捕捉。如果是因为Transact-SQL语句的执行而产生的错误,如键值冲突,使用第一种和第二种方法都可以捕捉,但是如果是逻辑错误,使用客户端代码进行捕捉就不太方便。因此,本文就如何使用Transact-SQL进行错误捕捉进行了讨论。

一、非致命错误(non-fatal error)的捕捉
    通过执行Transact-SQL而产生的错误可分为两种:致命错误(fatal error)和非致命错误(non-fatal error)。在Transact-SQL中只可以捕捉非致命错误(如键值冲突),而无法捕捉致命错误(如语法错误)。在Transact-SQL中可以通过系统变量@@ERROR判断最近执行的一条语句是否成功执行。如果发生了错误,@@Error的值大于0,否则值为0。下面举一个例子说明 @@ERROR的使用。
    假设有一个表table1,在这个表中有两个字段f1,f2。其中f1是主键。

    INSERT INTO table1 VALUES(1'aa')
    
INSERT INTO table1 VALUES(1'bb')  --这条语句将产生一个错误
    IF @@ERROR > 0
      
PRINT '键值冲突'

    当执行第二条语句时发生键值冲突错误,@@ERROR被赋为错误号2627,因此输出结果显示'键值冲突'。使用@@ERROR系统变量时需要注意, @@ERROR只记录最近一次执行的Transact-SQL语句所发生的错误,如果最近一次执行的Transact-SQL没有发生错误, @@ERROR的值为0。因此,只能在被捕捉的那条Transact-SQL语句后使用@@ERROR。
    在SQL Server中,不仅可以捕捉系统提供的错误,还可以自定义错误。有两种方法可以定义错误信息。
    1. 使用sp_addmessage系统存储过程添加错误信息,然后使用RAISERROR抛出错误。
    sp_addmessage将错误号,错误级别、错误描述等信息添加到系统表中,然后使用RAISERROR根据相应的错误号抛出错误信息。用户自定义的信息应该从50001开始。

   EXEC sp_addmessage @msgnum = 50001@severity = 16,
   
@msgtext = 'sql encounter an error(%s).',
   
@lang = 'us_english'

   
EXEC sp_addmessage @msgnum = 50001@severity = 16,
   
@msgtext = 'sql遇到了一个错误(%1!).'

   如果使用的SQL Server版本是非英语版本,在添加本地错误信息时必须首先添加英文的错误信息。错误描述可以象c语言中的printf的格式字符串一样使用参数,如% s、%d。但要注意的是在英文版的错误信息中要使用%s、%d等形式,而在本地化的错误信息中要使用%1!、%2!等形式,在每个%?(1 <= ? <= n)后需要加一个!,而且%?的数目必须和英文版的错误信息的参数一致。
   在未插入本地化错误信息时,RAISERROR将使用英文版的错误信息。当插入本地化错误信息时,RAISERROR使用本地化的错误信息。
   RAISERROR(50001, 16, 1, '测试')
  
   输出的结果:
   服务器: 消息 50001,级别 16,状态 1,行 1
   sql遇到了一个错误(测试).

   其中'测试'字符串通过%1传入本地化的错误描述字符串中。
 
    2. 直接使用RAISERROR将错误抛出。
    使用第一种方法虽然使Transact-SQL语句看上去更整洁(这种方法类似于在编程语言中使用常量定义错误信息,然后在不同的地方通过错误编号引用这些错误信息。),但是这样做却使错误信息和数据库的耦合度增加,因为如果将这些带有RAISERROR的Transact-SQL放到别的SQL Server数据库上执行,由于在其它的数据库中还未添加错误信息,因此会产生RAISERROR调用错误,除非使用sp_addmessage将所需的错误信息再加入到其它的数据库中。
    基于上述原因,RAISERROR不仅可以根据错误代码抛出错误信息,也可以直接通过错误描述格式字符串抛出错误信息。
    RAISERROR('sql遇到了一个错误(%s)', 16, 1, '测试')

二、逻辑错误的捕捉
    在实际应用中,更多的是由于某些业务要求而产生的逻辑错误。这些错误无法通过@@ERROR进行捕捉。如果使用客户端代码进行捕捉,那么Transact -SQL必须一条一条地执行。如果使用存储过程,那么发生在存储过程内部的逻辑错误就很难在客户端代码中进行捕捉,因此,下面将讨论如何使用 Transact-SQL捕捉逻辑错误。
    所谓逻辑错误,就是在执行完Transact-SQL后,执行结果与业务要求的结果不符而产生的。为了说明如何处理逻辑错误,我们再建立一个表 table2,这个表的结构和table1完全一样,只是f1字段不再是主键了。然后建立一个存储过程,它的功能是在table1和table2中同时插入一条记录,但是这条记录必须满足两个条件。
    1.f1值不能大于100。
    2.要插入的记录在table1中不存在,如果存在,在table1和table2中都不插入这条记录。

CREATE PROCEDURE p1(@Num int)
AS
DECLARE @Error int@RowCount int
BEGIN TRANSACTION
INSERT INTO table2 VALUES(@Num'p')
IF @Num > 100
BEGIN
  
RAISERROR('%s的值不能大于100。'161'@Num')   
  
ROLLBACK TRANSACTION
  
RETURN 1
END
ELSE
BEGIN
  
SELECT f1 FROM table1 WHERE f1 = @Num
  
IF @@ROWCOUNT > 0
  
BEGIN
    
RAISERROR('table1中已经存在%d了。'161@Num)
    
ROLLBACK TRANSACTION
    
RETURN 2
  
END
  
ELSE
  
BEGIN
    
INSERT INTO table1 VALUES(@Num'p')
    
COMMIT TRANSACTION
    
RETURN 0
  
END
END


    在这个存储过程中一开始使用BEGIN TRANSACTION显示地开始一个事务,然后当上述两种错误发生时使用ROLLBACK TRANSACTION恢复到初始状态,如果成功插入,使用COMMIT TRANSACTION提交改变。可以通过如下语句进行调用。

DECLARE @ErrNum int
EXEC @ErrNUm = p1 2
PRINT @ErrNum

    可以通过@ErrNum得到p1返回的错误代码,如果返回0,表示执行成功。

SQL Server2005中错误捕捉的新功能
    虽然在以前的SQL Server版本中可以通过一些技巧实现错误捕捉,但有时需要增加一些额外的开销,如在p1中使用了SELECT语句。庆幸的是在SQL Server2005中提供了和大多数编程语言类似的try...catch错误捕捉功能,从而使Transact-SQL第一次可以真正地进行错误捕捉。使用try...catch可以将p1的下半部分改写为如下形式。


ELSE
BEGIN
  
BEGIN TRY
    
INSERT INTO table1 VALUES(@Num'p')   
    
COMMIT TRANSACTION
    
RETURN 0
  
END TRY
  
BEGIN CATCH
    
RAISERROR('table1中已经存在%d了。'161@Num)
    
ROLLBACK TRANSACTION
    
RETURN 2
  
END CATCH
END

    可以看出,这个改写的部分未使用SELECT查询table1中是否已经有了某条记录,而是通过数据库的约束来进行判断的。如果键值冲突,就产生了错误,这样SQL语句就直接跳到BEGIN CATCH中执行错误处理代码。这样做效率要比上一个版本高得多,而且如果将RAISERROR去掉,p1就不会抛出任何错误,只是返回了一个错误码,这样有利于客户端代码进行处理。
    在Transact-SQL中进行错误捕捉,如果使用的是SQL Server2005,我的建议是尽量使用try...catch,因此它会捕捉到未预料到的错误,并且会使Transact-SQL更容易维护。当然,这样做就无法将Transact-SQL移植到SQL Server2000或更低的版本上运行,要是想写通用的Transact-SQL,还是使用传统的方法捕捉错误吧!

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/407480.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

ListT用法

List<T>和List的用法是一模一样的,不同的是,List支持任何类型而List<T>只支持T类型,T是在定义时指定的类型. 如: List<string> listnew List<string>; 这样的话,每个list里的元素都是字符串. 同理, List<int> listnew List<int>; 这样的话…

reduce

reduce reduce(callback,initiaValue)会传入两个变量&#xff0c;回调函数(callback)和初始值(initiaValue)。1.只传回调函数 prev只取数组的第一个元素 next从数组的第二个元素开始一直往下取var arr ["a", "b", "c", "d", "e&…

CSS之Screen视图属性

1.availWidth和availHeight availWidth和availHeight返回的是显示器可用宽高&#xff0c;注意不包括开始菜单栏这种东东的高度和宽度 2.colorDepth 表示显示器的颜色深度 3.width和height 表示显示器屏幕的宽高&#xff0c;和availWidth和availHeight的区别是返回的宽高包含…

[Linux C]递归遍历指定目录下的子目录和文件

/*功能&#xff1a;演示了在Linux下利用C语言递归遍历指定目录下的子目录(不含隐藏目录)和文件*/#include <stdio.h>#include <dirent.h>#include <string.h>void List(char *path){struct dirent *ent NULL;DIR *pDir;if((pDir opendir(path)) ! NULL){wh…

Java的多进程运行模式分析

本文曾发表于天极网&#xff1a;http://dev.yesky.com/284/2659284.shtml一般我们在java中运行其它类中的方法时&#xff0c;无论是静态调用&#xff0c;还是动态调用&#xff0c;都是在当前的进程中执行的&#xff0c;也就是说&#xff0c;只有一个java虚拟机实例在运行。而有…

HDU 2594 Simpsons’ Hidden Talents (字符串-KMP)

Simpsons’ Hidden Talents Problem DescriptionHomer: Marge, I just figured out a way to discover some of the talents we weren’t aware we had.Marge: Yeah, what is it?Homer: Take me for example. I want to find out if I have a talent in politics, OK?Marge: …

微信小程序几种常用弹窗提示

第一种&#xff1a;弹出提示框&#xff0c;可以选择确定或者取消。 代码&#xff1a;wx.showModal({title: 提示,content: 这是一个模态弹窗,success: function (res) {if (res.confirm) {//这里是点击了确定以后console.log(用户点击确定)} else {//这里是点击了取消以后conso…

[bash]删除文件中含特定字符串的行

]删除文件中含特定字符串的行[bash]: sed -e /abc/d a.txt // 删除a.txt中含"abc"的行&#xff0c;但不改变a.txt文件本身&#xff0c;操作之后的结果在终端显示 sed -e /abc/d a.txt > a.log // 删除a.txt中含"abc"的行&#xff0c;将操作之后的…

CSS之文档视图(DocumentView)和元素视图(ElementView)方法

1.elementFromPoint() 返回给定坐标处的元素。是个在目前而言&#xff0c;兼容性不错的方法 将给定位置处(100,100)的元素的字体颜色设置为红色 2.getBoundingClientRect() 返回的是返回的是一个对象&#xff0c;包含 top, left, right, 和 bottom四个属性值&#xff0c;大小…

谈CRM产品设计的指导思想

客户关系管理&#xff08;CRM&#xff09;其本意强调的是对客户“关系”进行有效管理&#xff0c;从而达到维持较高的客户占有率&#xff08;customershare&#xff09;的目的。所谓关系&#xff0c;是指两个事物之间其中的一方对另一方的行为方式以及感受状态。所以&#xff0…

try catch finally

<!DOCTYPE html> <html lang"en"><head><meta charset"UTF-8" /><title>try catch finally</title><script>// 异常最大的特征,就是一旦出现异常,后面的代码将不会再执行// 那为了保证后面的代码在出现异常之后…

Java笔记-IO流的运用

--如果朋友您想转载本文章请注明转载地址"http://www.cnblogs.com/XHJT/p/3877386.html "谢谢-- 1.InputStream和System.in&#xff08;Scanner&#xff09; InputStream 输出流以字节为单位来获取数据&#xff0c;且需要复杂的判断并创建字节数组作为缓冲 另外字节转…

[Bash]kill指定的进程名

通过bash来kill指定的进程名&#xff0c;bash文件名为&#xff1a;/home/zcm/bin/d.sh&#xff0c;内容如下&#xff1a; #!/bin/shif [ "$1" "" ]; thenecho "Usage: sh $0 <processname>"exit 0fi#s1ps -ef|grep $1|grep -v grep|awk …

CSS之元素视图属性

1.clientLeft和clientTop 返回的是内容区域的左上角相对于整个元素左上角的位置&#xff08;包括边框&#xff09; 2.clientWidth和clientHeight 表示内容区域的高度和宽度&#xff0c;包括padding大小&#xff0c;但是不包括边框和滚动条 3.offsetLeft和offsetTop offsetLe…

阿里巴巴指东打西,PC之后卖盒饭?

这两天&#xff0c;看到了数篇关于阿里巴巴的报道&#xff0c;核心内容都是阿里巴巴联手英特尔&#xff0c;发起主题为“助力中国”的电子商务推动计划&#xff0c;并将推出面向中小企业电子商务专用电脑的消息。从报道看&#xff0c;阿里巴巴将同英特尔合作&#xff0c;推出贴…

SQL-MSSQL-CODE大全

SqlServer数据库语句大全 /*********************************************************/ 目录清单CONTEXT LIST /*********************************************************/ 1.数据库DataBase 1.1数据库建立/删除create/drop database 1.2数据库备份与恢复backup/restore d…