25

我需要知道如何将 int 转换为可为空的 int。但是,我不断收到错误消息“没有为类型 'System.Nullable`1[System.Int32]' 和 'System.Int32' 定义二元运算符 Equal。” 任何解决方案。它需要是 Microsoft SQL Server 的可为空的 int 类型。

 somevalue = Expression.Constant(something.GetValue(some,null).To<Nullable<System.Int32>> ());

public static T To<T>(this object obj)
    {
        Type t = typeof(T);
        Type u = Nullable.GetUnderlyingType(t);

        if (u != null)
        {
            if (obj == null)
                return default(T);

            return (T)Convert.ChangeType(obj, u);
        }
        else
        {
            return (T)Convert.ChangeType(obj, t);
        }
    }'
4

6 回答 6

27

通常,您使用强制转换转换intan int?

int? myNullable = (int?) 15;
int myInt = (int) myNullable;
于 2013-08-02T15:08:22.767 回答
25

该代码似乎是您在给定一个不可为空类型的值时To尝试构造一个可空类型的值,但这根本不是解决此问题的正确方法。您尝试执行此操作的方式表明您对盒装值类型的工作方式存在误解。Constant

该错误消息表明您正在构建一个二元运算符表达式树节点,该节点具有一个可空 int 类型的表达式节点和一个 int 类型的表达式节点作为其操作数。那是不合法的;它们必须都是可为空的 int。您应该做的是将不可为空的 int 表达式树节点包装在表达式树节点中,该Convert 表达式树节点将其转换为可为空的 int,然后将其传递二元运算符表达式树节点构造函数。

也就是说,这是错误的:

var someIntExpr = Expression.Constant(123, typeof(int));
var someNubIntExpr = Expression.Constant(null, typeof(int?));
var badEq = Expression.Equal(someIntExpr, someNubIntExpr);

这是正确的:

var goodEq = Expression.Equal(Expression.Convert(someIntExpr, typeof(int?)),  someNubIntExpr);

那么为什么你做错了什么?

您有一个To<T>返回T. 它正确地接受int并返回等价​​的int?。那么呢?你将它传递给Expression.Constant,它将可为空的 int装箱成一个装箱的 int,然后从中生成一个常量。您认为存在盒装可为空值类型之类的东西,但实际上没有!可空值类型将绑定到空引用或绑定到不可空值类型。

所以你也可以通过不做任何这些疯狂的事情来解决你的问题。如果您手头有一个装箱的 int,并且需要一个可为空类型的常量表达式树节点,只需提供 type 即可

Expression.Constant(someBoxedIntValue, typeof(int?))

完毕。所以:总结一下,你有两个解决方案:

  • 如果您手头有一个装箱的 int,请将它和您想要的可为空值类型传递给Constant工厂,或者
  • 如果你手头有一个 int 类型的表达式节点,那么使用Convert表达式节点工厂,并将它和所需的类型传递给它。

两者都会为您返回一个正确类型的表达式节点,以便与另一个可为空的 int 进行比较。

于 2013-08-02T23:24:03.343 回答
5
int test = 0; // set int

int? num = test; // convert test to a nullable int

num = null; // set num as null
于 2013-08-02T15:07:35.480 回答
3
int i = 1;
int? k;
k = i as int?;

像这样,您会将 int 转换i为可为空的 int ;)

int?是 的简短版本Nullable<int>

于 2013-08-02T15:07:04.337 回答
3

像这样更简单的东西不起作用吗?

int i; 
int? temp = int.TryParse(<your value>, out i) ? (int?)i : null;
于 2013-08-02T15:11:31.473 回答
3

干得好。可空原始解决方案的通用字符串。

int? n = "  99 ".ToNullable<int>(); 

/// <summary>
/// Developed by Taylor Love
/// </summary>
public static class ToNullableStringExtension
{
    /// <summary>
    /// <para>More convenient than using T.TryParse(string, out T). 
    /// Works with primitive types, structs, and enums.
    /// Tries to parse the string to an instance of the type specified.
    /// If the input cannot be parsed, null will be returned.
    /// </para>
    /// <para>
    /// If the value of the caller is null, null will be returned.
    /// So if you have "string s = null;" and then you try "s.ToNullable...",
    /// null will be returned. No null exception will be thrown. 
    /// </para>
    /// </summary>
    /// <typeparam name="T"></typeparam>
    /// <param name="p_self"></param>
    /// <returns></returns>
    public static T? ToNullable<T>(this string p_self) where T : struct
    {
        if (!string.IsNullOrEmpty(p_self))
        {
            var converter = System.ComponentModel.TypeDescriptor.GetConverter(typeof(T));
            if (converter.IsValid(p_self)) return (T)converter.ConvertFromString(p_self);
            if (typeof(T).IsEnum) { T t; if (Enum.TryParse<T>(p_self, out t)) return t;}
        }

        return null;
    }

https://github.com/Pangamma/PangammaUtilities-CSharp/tree/master/src/StringExtensions

于 2017-02-24T18:00:57.813 回答