`
seara
  • 浏览: 655031 次
  • 性别: Icon_minigender_1
  • 来自: 广州
社区版块
存档分类
最新评论

在SQL Server2005中进行错误捕捉

阅读更多
本文为原创,如需转载,请注明作者和出处,谢谢!

本文曾发表于IT168:http://tech.it168.com/db/s/2006-07-21/200607211621906_1.shtml

任何程序都可能出现错误,在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是主键。

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->INSERTINTOtable1VALUES(1,'aa')
INSERTINTOtable1VALUES(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开始。

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->EXECsp_addmessage@msgnum=50001,@severity=16,
@msgtext='sqlencounteranerror(%s).',
@lang='us_english'

EXECsp_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中都不插入这条记录。

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->CREATEPROCEDUREp1(@Numint)
AS
DECLARE@Errorint,@RowCountint
BEGINTRANSACTION
INSERTINTOtable2VALUES(@Num,'p')
IF@Num>100
BEGIN
RAISERROR('%s的值不能大于100。',16,1,'@Num')
ROLLBACKTRANSACTION
RETURN1
END
ELSE
BEGIN
SELECTf1FROMtable1WHEREf1=@Num
IF@@ROWCOUNT>0
BEGIN
RAISERROR('table1中已经存在%d了。',16,1,@Num)
ROLLBACKTRANSACTION
RETURN2
END
ELSE
BEGIN
INSERTINTOtable1VALUES(@Num,'p')
COMMITTRANSACTION
RETURN0
END
END


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

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->DECLARE@ErrNumint
EXEC@ErrNUm=p12
PRINT@ErrNum

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

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

<!--<br /> <br /> Code highlighting produced by Actipro CodeHighlighter (freeware)<br /> http://www.CodeHighlighter.com/<br /> <br /> -->
ELSE
BEGIN
BEGINTRY
INSERTINTOtable1VALUES(@Num,'p')
COMMITTRANSACTION
RETURN0
ENDTRY
BEGINCATCH
RAISERROR('table1中已经存在%d了。',16,1,@Num)
ROLLBACKTRANSACTION
RETURN2
ENDCATCH
END

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



国内最棒的Google Android技术社区(eoeandroid),欢迎访问!

《银河系列原创教程》发布

《Java Web开发速学宝典》出版,欢迎定购

分享到:
评论

相关推荐

    sql server数据库中raiserror函数用法的详细介绍

    sql server数据库中raiserror函数的用法 server数据库中raiserror的作用就和asp.NET中的throw new Exception一样,用于抛出一个异常或错误。这个错误可以被程序捕捉到。 raiserror的常用格式如下: raiserror(‘错误...

    SQLServer2005及SQLServer2008快照同步配置说明汇编.pdf

    SQL Server 2005和SQL Server 2008中的快照同步是一种数据库复制技术,主要用于在不同的数据库实例间同步数据,确保数据的一致性和完整性。快照同步配置涉及多个步骤,包括安装SQL Server、配置服务、创建数据库、...

    SQL.Server性能优化工具Profiler

    其中,Profiler 是 SQL Server 中的一个重要的性能优化工具,能够帮助用户捕捉和分析数据库中的性能问题。 Profiler 是一个图形化的工具,能够实时地捕捉和显示数据库中的各种事件,包括查询、锁定、死锁、错误等等...

    SQLServer恢复表级数据详解

    在使用SQLServer数据库时,数据的备份和恢复是数据库管理中非常重要的环节。尤其在开发和生产环境中,表级数据的恢复显得尤为重要,因为关键表往往存放着核心数据,一旦出现数据损坏,需要迅速采取措施进行恢复。...

    SQL Server 2005图书管理系统课程设计.doc

    在这个图书管理系统课程设计中,我们将深入探讨如何利用SQL Server 2005构建高效、稳定且功能丰富的系统。 首先,理解SQL Server 2005的基础架构至关重要。它基于关系型数据库模型,支持SQL语言进行数据操作,包括...

    Inside Microsoft SQL Server2005_ T-SQL Programming.rar

    《Inside Microsoft SQL Server 2005: T-SQL Programming》是针对微软SQL Server 2005数据库管理系统中T-SQL(Transact-SQL)编程的一本详细指南。这本书深入探讨了SQL Server 2005的核心组件,特别是T-SQL在数据...

    SqlServer查询当下数据库还原状态

    在SQL Server环境中,了解当前正在进行的数据库还原操作的状态对于DBA(数据库管理员)或开发人员来说至关重要。这不仅能帮助他们监控还原进度,还能确保数据库在还原过程中不会出现异常中断或其他问题。下面将详细...

    Microsoft PHP Driver for SQLServer

    处理错误和警告部分会介绍如何捕捉和处理在与SQL Server交互过程中可能出现的错误和警告信息。 日志活动部分则会指导如何记录与数据库交互的活动日志,以用于调试和性能监控。 常量部分会介绍Microsoft Drivers ...

    java调用sqlserver存储过程.pdf

    在 SQL Server 中,存储过程是一个预编译的 SQL 语句集,可以对数据库中的数据进行操作。在本例中,创建了一个名为 `InsertUser` 的存储过程,该存储过程用于向 `BookUser` 表中插入新用户信息。 2. 存储过程的参数...

    .NET C# 连接SqlServer类

    6. 异常处理:在这个例子中,使用 try-catch 语句来处理异常,可以捕捉和处理程序中的错误。 7. 使用委托和事件:在这个例子中,使用委托和事件来处理程序中的事件,可以将事件处理程序与事件源分离。 8. 使用命名...

    asp.net+sqlserver2005

    9. **错误处理和日志记录**:理解如何在ASP.NET中捕捉和处理异常,以及记录应用程序日志。 10. **性能优化**:可能涉及到数据库查询优化、减少服务器往返次数、缓存策略等。 通过这些实际项目,开发者不仅可以学习...

    delphi_sql.rar_DelPhi SQl_delphi sql server_sql delphi

    描述中提到的"delphi sql server 2005 control and samples program"表明,这个资源可能包含了如何在Delphi中控制SQL Server 2005的指导,以及一些实际的示例程序。这些示例可能涵盖了连接数据库、执行查询、处理...

    Microsoft SQL Server Performance tunning with DMV

    - 查看Windows应用程序日志(Windows)和SQL Server错误日志(SQL Server Management Studio):通过查看应用程序日志和错误日志可以追踪到数据库性能的问题和错误信息。 - 保存和分析死锁图形(SQL Server Profiler...

    SQL Server服务器同步备份.rar_SQL 服务_SQL同步_sql server 备份_同步 SQL

    在SQL Server中,备份是数据保护的重要环节,它可以防止硬件故障、软件错误或人为错误导致的数据丢失。SQL Server 提供了多种备份类型,包括完整备份、差异备份和事务日志备份,以满足不同级别的恢复需求。 SQL同步...

    PHP5.2.3连接 SQLSERVER 注意事项

    在进行连接时,需要确认SQL Server的版本与PHP的版本相匹配,以避免兼容性问题。 2. **SQL Server扩展**: PHP连接SQL Server通常依赖于`sqlsrv`或`mssql`扩展。由于PHP 5.2.3较旧,`sqlsrv`扩展可能尚未包含,...

    JDBC连接SQL Server2008插入、修改、删除、查询代码

    ### JDBC 连接 SQL Server 2008 进行数据操作详解 #### 一、简介 在 Java 开发环境中,使用 JDBC (Java Database ...此外,在实际开发过程中还需要注意错误处理和资源管理等方面,以确保程序的健壮性和稳定性。

    SQL Server 2005 中使用 Try Catch 处理异常

    在SQL Server 2005中,异常处理机制得到了显著改进,引入了TRY...CATCH结构,这使得处理错误的方式更加类似于高级编程语言如C#和Java。在SQL Server 2000中,异常处理主要依赖于@@ERROR变量,但这种方式存在局限性,...

    安装SQL SERVER2000弹不出安装界面

    在安装Microsoft SQL Server 2000的过程中,有时可能会遇到无法正常启动安装界面的问题。这个问题通常是由于多种原因造成的,如系统兼容性、缺少组件、已有的SQL Server实例冲突或者安装程序执行过程中的某些错误。...

    SQL Server中OLE对象开发及应用.pdf

    而OLE对象在SQL Server中的应用,主要体现在扩展SQL Server与外部系统交互的能力上,使得数据库能够利用外部资源进行数据处理和信息管理。 ### 关键知识点一:SQL Server与OLE对象交互能力的增强 在早期版本的SQL ...

Global site tag (gtag.js) - Google Analytics