C#ADO.NET:null和DbNull -是否有更有效的语法?

我有一个DateTime?,我正在尝试使用dataPrm.Value插入到字段中。我正在像这样创建参数:

DbParameter datePrm = updateStmt.CreateParameter();
datePrm.ParameterName = "@change_date";

然后我想将DateTime?的值放到dataPrm.Value中,同时考虑nulls。

我以为最初我会很聪明:

datePrm.Value = nullableDate ?? DBNull.Value;

但这失败并显示错误

运算符“ ??” 不能应用于类型为'System.DateTime?'的操作数 和“ System.DBNull”

因此,我猜想只有在第二个参数是第一个参数的非空版本时,该方法才有效。 所以我去了:

datePrm.Value = nullableDate.HasValue ? nullableDate.Value : DBNull.Value;

但这也不起作用:

无法确定条件表达式的类型,因为“ System.DateTime”和“ System.DBNull”之间没有隐式转换

但是我不想在这些类型之间转换!

到目前为止,我唯一可以工作的是:

if (nullableDate.HasValue)
  datePrm.Value = nullableDate.Value;
else
  datePrm.Value = DBNull.Value;

那真的是我写这个的唯一方法吗? 有没有一种方法可以使用三元运算符来工作?

更新:我真的不明白为什么? 版本不起作用。 MSDN说:

?? 如果运算符不为null,则它返回左侧的操作数,否则返回右侧的操作数。

那正是我想要的!

Update2:嗯,这在最后很明显:

datePrm.Value = nullableDate ?? (object)DBNull.Value;
Stewart Johnson asked 2020-02-13T22:36:52Z
6个解决方案
62 votes

啊哈! 我发现了比@Trebz的解决方案更有效的解决方案!

datePrm.Value = nullableDate ?? (object)DBNull.Value;
Stewart Johnson answered 2020-02-13T22:37:05Z
6 votes

如果您使用的是SQLServer,则System.Data.SqlTypes命名空间包含一些避免烦人的类型转换的实用程序类。 例如,代替此:

var val = (object) "abc" ?? DBNull.Value;

你可以这样写:

var val = "abc" ?? SqlString.Null;
Gian Marco Gherardi answered 2020-02-13T22:37:30Z
5 votes

如果您使用它会工作

datePrm.Value = nullableDate.HasValue ? (object)nullableDate.Value : DBNull.Value;
dnolan answered 2020-02-13T22:37:50Z
5 votes

如果您使用的是C#3.0,则可以创建扩展方法来轻松实现此目的:

public static class DBNullableExtensions
{
    public static object ToDBValue<T>(this Nullable<T> value) where T:struct
    { 
        return value.HasValue ? (object)value.Value : DBNull.Value;
    }
}


class Program
{
    static void Main(string[] args)
    {
        int? x = null;

        Console.WriteLine(  x.ToDBValue() == DBNull.Value );
    }
}
Pop Catalin answered 2020-02-13T22:38:10Z
1 votes

我认为您第二次尝试的错误是由于nullableDate.Value和DBNull.Value是不同的类型,并且三元运算符在两种情况下都需要选择一种类型才能返回。 我没有测试的环境,但我认为这应该对您有用:

datePrm.Value = nullableDate.HasValue ? (object)nullableDate.Value : (object)DBNull.Value;
Dan answered 2020-02-13T22:38:30Z
0 votes

我这样做的方法是,我有一个静态实用程序类,该类仅经过检查并检查参数值是否为null,然后将值设置为DBNull。 我只是在致电执行之前就这样做了。

Darren Kopp answered 2020-02-13T22:38:50Z
translate from https://stackoverflow.com:/questions/218808/c-sharp-ado-net-nulls-and-dbnull-is-there-more-efficient-syntax