设为首页 收藏本站
查看: 645|回复: 0

[经验分享] 谈谈基于SQL Server 的Exception Handlingp[下篇]

[复制链接]

尚未签到

发表于 2015-6-29 10:38:10 | 显示全部楼层 |阅读模式
六、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方法:
  

DSC0000.gif public static  bool CreateUser(string userName)
DSC0001.gif DSC0002.gif          DSC0003.gif {
DSC0004.gif             string procedureName = "P_USERS_I";
            Dictionary parameters = new Dictionary();
            parameters.Add("user_id", Guid.NewGuid().ToString());
            parameters.Add("user_name", userName);
            try
DSC0005.gif DSC0006.gif             {
                ExecuteCommand(procedureName, parameters);
                return true;
DSC0007.gif             }
            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;
            }
DSC0008.gif         }
  在Main()中调用这个CreateUser():

   DSC0009.gif

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


SET @error_message    = ERROR_MESSAGE()
SET @error_serverity    =ERROR_SEVERITY()
SET @error_state    = ERROR_STATE()
RAISERROR(@error_message, @error_serverity,@error_state)  
  将会得到这样的结果:

   DSC00010.gif

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

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!'))
  

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()种添加了下面一段代码:
  

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!"

DSC00011.gif

[原创]谈谈基于SQL Server的Exception Handling - PART I
[原创]谈谈基于SQL Server 的Exception Handling - PART II
[原创]谈谈基于SQL Server 的Exception Handling - PART III

运维网声明 1、欢迎大家加入本站运维交流群:群②:261659950 群⑤:202807635 群⑦870801961 群⑧679858003
2、本站所有主题由该帖子作者发表,该帖子作者与运维网享有帖子相关版权
3、所有作品的著作权均归原作者享有,请您和我们一样尊重他人的著作权等合法权益。如果您对作品感到满意,请购买正版
4、禁止制作、复制、发布和传播具有反动、淫秽、色情、暴力、凶杀等内容的信息,一经发现立即删除。若您因此触犯法律,一切后果自负,我们对此不承担任何责任
5、所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其内容的准确性、可靠性、正当性、安全性、合法性等负责,亦不承担任何法律责任
6、所有作品仅供您个人学习、研究或欣赏,不得用于商业或者其他用途,否则,一切后果均由您自己承担,我们对此不承担任何法律责任
7、如涉及侵犯版权等问题,请您及时通知我们,我们将立即采取措施予以解决
8、联系人Email:admin@iyunv.com 网址:www.yunweiku.com

所有资源均系网友上传或者通过网络收集,我们仅提供一个展示、介绍、观摩学习的平台,我们不对其承担任何法律责任,如涉及侵犯版权等问题,请您及时通知我们,我们将立即处理,联系人Email:kefu@iyunv.com,QQ:1061981298 本贴地址:https://www.yunweiku.com/thread-81454-1-1.html 上篇帖子: SQL Server 错误日志收缩(ERRORLOG) 下篇帖子: 谈谈基于SQL Server 的Exception Handling[上篇]
您需要登录后才可以回帖 登录 | 立即注册

本版积分规则

扫码加入运维网微信交流群X

扫码加入运维网微信交流群

扫描二维码加入运维网微信交流群,最新一手资源尽在官方微信交流群!快快加入我们吧...

扫描微信二维码查看详情

客服E-mail:kefu@iyunv.com 客服QQ:1061981298


QQ群⑦:运维网交流群⑦ QQ群⑧:运维网交流群⑧ k8s群:运维网kubernetes交流群


提醒:禁止发布任何违反国家法律、法规的言论与图片等内容;本站内容均来自个人观点与网络等信息,非本站认同之观点.


本站大部分资源是网友从网上搜集分享而来,其版权均归原作者及其网站所有,我们尊重他人的合法权益,如有内容侵犯您的合法权益,请及时与我们联系进行核实删除!



合作伙伴: 青云cloud

快速回复 返回顶部 返回列表