7

我使用 System.Data.Common.DbCommand 将新行插入数据库。关键是,这一行已经存在。

try
{
    [...]
    DbCommand insertCommand = [...]
    insertCommand.ExecuteScalar();
    [...]
}
catch (System.Exception exception)
{
   [...]
   throw;
}

如果我要明确捕获 System.Data.SqlClient.SqlException 我可以评估 ErrorNumber 如下所示。

try
{
  //insertion code
}
catch(SqlException ex)
{
  if(ex.Number == 2627)
  {
     //Violation of primary key. Handle Exception
  }
}

SqlException 上下文中的 ErrorNumber 2627 表示违反唯一键。请参阅https://msdn.microsoft.com/en-us/library/ms151757%28v=sql.110%29.aspx

到目前为止,一切都很好。因为我正在使用 DbCommand 并因此使用不同类型的关系数据库管理系统,所以我正在寻找一种更通用的方式来捕获这种违反唯一键约束的行为。

感谢帮助。

4

2 回答 2

3

恐怕没有内置的通用解决方案。作为一种解决方法,您可以为以下内容创建一个包装器Execute...

public static int ExecuteWithNiceExceptions(this IDbCommand cmd)
{
    try
    {
        return cmd.ExecuteNonQuery();
    }
    catch(SqlException ex)
    {
        if (ex.Number == 2627)
        {
            throw new PrimaryKeyViolationException(cmd, ex);
        }
        else
        {
            throw;
        }
    }
    catch (OleDbException ex)
    {
        ... 
    }
    ...
}

这样,您可以将不同的、特定于实现的异常“转换”为通用的、有意义的异常。

于 2015-03-10T10:43:04.347 回答
0

您应该允许数据库分配密钥或使用 GUID 作为密钥,因为数据库是围绕多用户同时访问而设计的,您的代码无法知道下一个密钥应该是什么,因此您需要让数据库管理自己的密钥或使用保证唯一性的密钥。

但是,由于每个数据库实现都使用自己的一组错误代码,因此在不知道您使用的是哪个提供商的情况下,无法赋予错误代码含义,除非向 ISO 请求他们创建错误代码的国际标准,我对此表示怀疑将工作

于 2015-03-10T10:46:07.240 回答