C# 如何解决 Func<T, bool> 中的逆变问题?

标签 c# generics lambda casting contravariance

假设我有以下代码:

class Program
{
    static IList<Func<object, bool>> _exclusions = new List<Func<object, bool>>();

    static void Main(string[] args)
    {       
        SetExclusions<PropertyInfo>(x => typeof(ValueType).IsAssignableFrom(x.PropertyType));           
    }

    public static void SetExclusions<T>(Func<T, bool> lambda)
        where T : class
    {
        Func<object, bool> l = (Func<object, bool>)lambda; // <= InvalidCastException
        _exclusions.Add(l);
    }
}

当然它不起作用,因为 Func 中的 T 类型参数是逆变的 - 我不能将 PropertyInfo 作为比 object.

有没有办法绕过它?从我的角度来看,之后做这样的事情是完全正确的:

foreach (var e in GetExclusions<PropertyInfo>())
{
    var a = members.Where(e);    
}

最佳答案

在提出任何建议之前,当前的设计很可能不适合您需要解决的问题。所以我真的建议只要有机会就重新访问它

现在针对您的问题,一种选择是将对象转换为 lambda 中的预期类型; 像这样:

Func<object, bool> l = o => lambda((T) o); 

然后您可以在 Dictionary<Type, Func<object, bool>> 中跟踪什么类型的表达式适用于每种不同的 Type .如果您不想保留字典,可能还有其他选择,但它们会涉及使用包含相同信息的类

关于C# 如何解决 Func<T, bool> 中的逆变问题?,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/32649441/

相关文章:

c# - 如何从接口(interface)引用实现类?

c# - 两个 XML 的比较 - "illegal characters in path"

c# - 使用 UI 进行后台工作

c# - 如何设置泛型类的基类属性

c# - 在 MVC 中拥有仅数据 Controller 的良好做法?

java - 静态方法引用泛型类型解决方案

c# - 基于类型重载泛型类的构造函数

c++ - 调用的对象类型 'auto' 不是函数或函数指针

c++ - 尝试传递一个 constexpr lambda 并使用它来显式指定返回类型

c# - 我如何将代码更改为linq样式