2

我对这个错误感到很困惑:

Cannot implicitly convert type 'System.Func<T,T,T> [c:\Program Files (x86)\Reference Assemblies\Microsoft\Framework\.NETFramework\v4.0\mscorlib.dll]' to 'System.Func<T,T,T> [c:\Program Files (x86)\Reference Assemblies\Microsoft\Framework\.NETFramework\v4.0\mscorlib.dll]' path\to\my\project\Operators.cs

类型是相同的,为什么它甚至试图做一个演员?这是代码:

public static class Operators<T>
{
    private static Func<T,T,T> _add = null;

    public static T Add<T>(T a, T b)
    {
        if (_add == null) {
            var param1Expr = Expression.Parameter(typeof (T));
            var param2Expr = Expression.Parameter(typeof (T));
            var addExpr = Expression.Add(param1Expr, param2Expr);
            var expr = Expression.Lambda<Func<T, T, T>>(addExpr, param1Expr, param2Expr);
            _add = expr.Compile(); // <--- error occurs here
        }
        return _add.Invoke(a, b);
    }
}
4

2 回答 2

13

问题是您的方法是通用的,引入了一个新的类型参数T。所以T外部方法与T内部方法不同。

只需将您的方法更改为不是通用的:

public static T Add(T a, T b)

……应该没问题。

更清楚地说,您的代码当前等效于:

public static class Operators<TC>
{
    private static Func<TC, TC, TC> _add = null;

    public static TM Add<TM>(TM a, TM b)
    {
        if (_add == null) {
            var param1Expr = Expression.Parameter(typeof(TM));
            var param2Expr = Expression.Parameter(typeof(TM));
            var addExpr = Expression.Add(param1Expr, param2Expr);
            var expr = Expression.Lambda<Func<TM, TM, TM>>
                          (addExpr, param1Expr, param2Expr);
            _add = expr.Compile();
        }
        return _add.Invoke(a, b);
    }
}

请注意我如何将T引入的重命名为,将方法引入的重命名为。错误消息现在看起来更合理:TCTTM

Test.cs(19,20): error CS0029: Cannot implicitly convert type
        'System.Func<TM,TM,TM>' to 'System.Func<TC,TC,TC>'
于 2013-02-24T19:52:53.000 回答
2

Tfor 你的类Operators<T>Ttype 参数Add是不同的类型,所以不能保证这些类型是兼容的。

例如你可以这样做:

Operators<string>.Add<int>(1, 2);

编译器会为此发出警告:

类型参数 'T' 与外部类型 'Operators' 中的类型参数同名

于 2013-02-24T19:53:00.673 回答