我有一些代码如下所示。首先,我有一些域类和一些特殊的比较器。
public class Fruit {
public int Calories { get; set; }
public string Name { get; set; }
}
public class FruitEqualityComparer : IEqualityComparer<Fruit> {
// ...
}
// A basket is just a group of Fruits.
public class BasketEqualityComparer : IEqualityComparer<IEnumerable<Fruit>> {
// ...
}
接下来,我有一个名为ConstraintChecker
的辅助类。它有一个简单的BaseEquals
方法,可以确保考虑一些简单的基本情况:
public static class ConstraintChecker {
public static bool BaseEquals(T lhs, T rhs) {
bool sameObject = l == r;
bool leftNull = l == null;
bool rightNull = r == null;
return sameObject && !leftNull && !rightNull;
}
还有一个SemanticEquals
方法,它只是一个BaseEquals
检查和你指定的比较器函数。
public static bool SemanticEquals<T>(
T lhs, T rhs, Func<T, T, bool> f) {
return BaseEquals(lhs, rhs) && f(lhs, rhs);
}
最后有一个SemanticSequenceEquals
方法接受两个IEnumerable<T>
个实例进行比较,一个IEqualityComparer实例将通过Enumerable.SequenceEquals
在列表中的每对元素上调用。
public static bool SemanticSequenceEquals<T, U, V>(U lhs,
U rhs,
V comparator)
where U : IEnumerable<T>
where V : IEqualityComparer<T> {
return SemanticEquals(lhs, rhs, (l, r) => lhs.SequenceEqual(rhs, comparator));
}
} // end of ConstraintChecker
SemanticSequenceEquals
的要点是,无论何时想要比较IEnumerable<T>
和T
个实例,都不必定义两个比较器;现在您只需指定一个IEqualityComparer<T>
,它也会在您调用SemanticSequenceEquals
时处理列表。所以我可以摆脱BasketEqualityComparer类,这很不错。
但是有一个问题。当您调用SemanticSequenceEquals时,C#编译器无法确定所涉及的类型:
// Error! Compiler can't infer the type parameters.
return ConstraintChecker.SemanticSequenceEquals(lhs, rhs,
new FruitEqualityComparer());
如果我明确指定它们,它可以工作:
return ConstraintChecker.SemanticSequenceEquals<Fruit, IEnumerable<Fruit>,
IEqualityComparer<Fruit>> (lhs, rhs, new FruitEqualityComparer());
显然,这是一个巨大的麻烦,并不是很干。我可以在这里更改什么,以便我不必显式写入类型参数?
答案 0 :(得分:2)
尝试仅指定T而不使用U和V。
public static bool SemanticSequenceEquals<T>(
IEnumerable<T> lhs,
IEnumerable<T> rhs,
IEqualityComparer<T> comparator)
{
return SemanticEquals(lhs, rhs, (l, r) => lhs.SequenceEqual(rhs, comparator));
}
答案 1 :(得分:1)
您可以显式键入SemanticSequenceEquals的参数。这对我来说似乎编译得很好:
public static bool SemanticSequenceEquals<T>(IEnumerable<T> lhs, IEnumerable<T> rhs, IEqualityComparer<T> comparator)
{
return SemanticEquals(lhs, rhs, (l, r) => lhs.SequenceEqual(rhs, comparator));
}
List<Fruit> a, b;
return ConstraintChecker.SemanticSequenceEquals(a, b, new FruitEqualityComparer());