前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >谈谈基于SQL Server 的Exception Handlingp[下篇]

谈谈基于SQL Server 的Exception Handlingp[下篇]

作者头像
蒋金楠
发布2022-05-09 12:56:30
2940
发布2022-05-09 12:56:30
举报
文章被收录于专栏:大内老A大内老A

六、SqlException

在上面一节中,我给出了一个完整的例子说明了:如何在将message定义在sys.messages中保证message的一致性和可维护性;如何在Stored procedure中使用RAISERROR将一个可预知的Error抛出;如何在Stored procedure中使用TRY/CATCH进行异常的捕捉;在Application如果处理从SQL Server抛出的Exception。实际上,SQL Server database Engine抛出、被我们的.NET最终捕获的SqlException,我们通过SqlException的属性可以得到Error的相关信息。下面是SqlException的属性列表:

  • public SqlErrorCollection Errors { get; }
  • public int LineNumber { get; }
  • public int Number { get; }
  • public string Procedure { get; }
  • public string Server { get; }
  • public override string Source { get; }
  • public byte State { get; }

有了前面的内容作铺垫,相信大家都知道每个属性分别表示的什么了吧。为了使大家对

stored procedure的Error和ADO.NET捕获的Error的Mapping有一个更加清晰的认识。我们来写一个Sample,我们沿用Create User的例子:

在stored procedure中,遇到重名通过RAISERROR抛出异常[在整篇文章中,使用到Error和Exception,大家可以看成是等效的]:     · Error Number:50001     · Severity:16     · State:1     · Message:This user is already existent

我们来修正一下CreateUser方法:

代码语言:javascript
复制
public static  bool CreateUser(string userName)
        {
            string procedureName = "P_USERS_I";
            Dictionary<string, object> parameters = new Dictionary<string, object>();
            parameters.Add("user_id", Guid.NewGuid().ToString());
            parameters.Add("user_name", userName);
            try
            {
                ExecuteCommand(procedureName, parameters);
                return true;
            }
            catch (SqlException ex)
            {
                Console.WriteLine("ex.Class\t: {0}", ex.Class);
                Console.WriteLine("ex.ErrorCode\t: {0}", ex.ErrorCode);
                Console.WriteLine("ex.LineNumber\t: {0}", ex.LineNumber);
                Console.WriteLine("ex.Message\t: {0}", ex.Message);
                Console.WriteLine("ex.Number\t: {0}", ex.Number);
                Console.WriteLine("ex.Procedure\t: {0}", ex.Procedure);
                Console.WriteLine("ex.Server\t: {0}", ex.Server);
                Console.WriteLine("ex.Source\t: {0}", ex.Source);
                Console.WriteLine("ex.State\t: {0}", ex.State);
         return false;
            }
        }

在Main()中调用这个CreateUser():

在这里我想特别说明一下SqlException.Number这个属性,它代表Database中的Error number[或者是@@ERROR、imessage_id],不过当我们使用RAISERROR语句,如果我们指定的一个表示error message的字符串,ADO.NET捕获的SqlException.Number这个属性默认为50000。比如我们将Error number换成error message:

代码语言:javascript
复制
SET @error_message    = ERROR_MESSAGE()
SET @error_serverity    =ERROR_SEVERITY()
SET @error_state    = ERROR_STATE()
RAISERROR(@error_message, @error_serverity,@error_state)

将会得到这样的结果:

还有一点需要特别提醒得是,我们可以在调用RAISERROR加了一个WITH SETERROR重句,强制将当前@@ERROR的值返回到客户端:

代码语言:javascript
复制
RAISERROR(@error_message,@error_serverity,@error_state) WITH SETERROR

七、 InfoMessage

上面的所以内容都围绕一个Exception handling的主题,在文章最后一部分我们想想一个和非Exception handling但是又和上面的内容很相关的主题:在Database通过Print语句输出的Message如何向Application传递。

在上面的例子中,有一个P_CLEAR_DATA的stored procedure,用于数据的清理。在操作结束后,有一个Print语句(PRINT ('All data have been deleted!'))

代码语言:javascript
复制
CREATE Procedure P_CLEAR_DATA
AS

    DELETE FROM dbo.T_USERS_IN_ROLES
    DELETE FROM dbo.T_USERS
    DELETE FROM dbo.T_ROLES
    
    PRINT ('All data have been deleted!')
    
GO

我们的现在的目标是在Application中,如何得到这段Message。要做到这点很简单,只需要用到SqlConnection的InfoMessage事件,当通过DbCommand执行上面一段Sql的时候,Print语句的执行将出发该事件。我们现在要做的就是注册这个事件,比如下面我们在ExecuteCommand()种添加了下面一段代码:

代码语言:javascript
复制
SqlConnection sqlConnection = connection as SqlConnection;
 if (sqlConnection != null)
    {
          sqlConnection.InfoMessage += delegate(object sender, SqlInfoMessageEventArgs e)
          {
                        Console.WriteLine(e.Message);
           };
     }

当我们调用Utility.Clear()的时候,就会输出"All data have been deleted!"

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2007-12-17,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 七、 InfoMessage
相关产品与服务
数据库
云数据库为企业提供了完善的关系型数据库、非关系型数据库、分析型数据库和数据库生态工具。您可以通过产品选择和组合搭建,轻松实现高可靠、高可用性、高性能等数据库需求。云数据库服务也可大幅减少您的运维工作量,更专注于业务发展,让企业一站式享受数据上云及分布式架构的技术红利!
领券
问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档