c# – 帮助C#generics错误 – “类型’T’必须是不可为空的值类型”

问题:

我是C#的新手,不明白为什么以下代码不起作用。

public static Nullable<T> CoalesceMax<T>(Nullable<T> a, Nullable<T> b) where T : IComparable
{
    if (a.HasValue && b.HasValue)
        return a.Value.CompareTo(b.Value) < 0 ? b : a;
    else if (a.HasValue)
        return a;
    else
        return b;
}

// Sample usage:
public DateTime? CalculateDate(DataRow row)
{
    DateTime? result = null;
    if (!(row["EXPIRATION_DATE"] is DBNull))
        result = DateTime.Parse((string)row["EXPIRATION_DATE"]);
    if (!(row["SHIPPING_DATE"] is DBNull))
        result = CoalesceMax(
            result
            DateTime.Parse((string)row["SHIPPING_DATE"]).AddYears(1));
    // etc.
    return result;
}

它在编译期间出现以下错误:


类型’T’必须是非可空值类型,以便在泛型类型或方法’System.Nullable <T>’中将其用作参数’T’

I’m new to C# and don’t understand why the following code doesn’t work.It gives the following error during compilation:The type ‘T’ must be a non-nullable value type in order to use it as parameter ‘T’ in the generic type or method ‘System.Nullable<T>’

解决方案:

方案1:

您需要添加T:struct约束:

public static Nullable<T> CoalesceMax<T>
    (Nullable<T> a, Nullable<T> b) where T : struct, IComparable

否则C#将尝试找出Nullable<T>含义,并意识到它还没有Nullable<T>本身所需的约束。 换句话说,您可以尝试调用:

CoalesceMax<string>(...)

这是没有意义的,因为Nullable<string>无效。

方案2:

Nullable<T>类型对它有约束,要求T为值类型(C#中的struct )。 这就是编译器告诉你Nullable<T>而不是你的函数或该函数的调用站点的原因 – 它是Nullable类,它是错误的根本原因,所以如果编译器指向的话,这实际上更有帮助对你的功能说“这是不对的,修好它!” (想象一下,如果CoalesceMax使用了几个泛型,并且只违反了其中一个,那么知道哪个泛型的约束被破坏比知道CoalesceMax中的一个或多个约束被破坏更有用。

解决方案是通过引入相同的约束来使您的T和它们的T兼容。 这是通过添加struct约束来完成的, struct约束必须在所有接口/新约束之前:

public static Nullable<T> CoalesceMax<T>(Nullable<T> a, Nullable<T> b) where T : struct, IComparable{
  ...
}

方案3:

您的通用方法使用Nullable<T>

但是,你没有约束T的类型,所以它最终可能是Nullable<Form> ,这显然是无效的。

您需要将约束更改为where T : struct, IComparable以确保T只能是值类型。

方案4:

不完全是OP的答案,但由于这是谷歌首次出现相同的错误消息,我不得不在我的类定义上添加约束,而不是我的方法,例如

public class MyClass<T> where T : struct
{
    public void MyMethod(T? value)
    {
    }
}

参考连接:

https://stackoverflow.com/questions/2230657/2230657