作为一个爱好项目(并且让自己更深入地沉浸在泛型/扩展方法中),我正在编写一个参数检查库!
我有一个名为 Argument 的模型,它描述了一个参数,如下所示:
public class Argument<T>
{
internal Argument(string name, T value)
{
Name = name;
Value = value;
}
public string Name { get; private set; }
public T Value { get; private set; }
}
当对参数进行验证时,会创建该对象的一个实例,并通过调用与其相关的扩展方法(包含实际逻辑)来执行单独的验证。
一种这样的扩展方法验证一个集合是否包含至少一个项目,目前看起来像这样:
public static Argument<IEnumerable<T>> HasItems<T>(this Argument<IEnumerable<T>> argument)
{
if (!argument.Value.Any())
throw Error.Generic(argument.Name, "Collection contains no items.");
return argument;
}
但它似乎不起作用。比如说,如果我要编写这个单元测试:
[TestMethod]
public void TestMethod1()
{
var argument = new List<int>() { 1, 2, 6, 3, -1, 5, 0 };
Validate.Argument("argument", argument)
.IsNotNull()
.HasItems()
.All(v => v.IsGreaterThan(0));
}
HasItems 没有出现在 Intellisense 中,我得到这个编译错误:
'Validation.Argument<System.Collections.Generic.List<int>>'
不包含“HasItems”的定义,并且找不到接受第一个类型参数的扩展方法“HasItems”'Validation.Argument<System.Collections.Generic.List<int>>'
(您是否缺少 using 指令或程序集引用?)
如果我尝试将值直接传递给扩展方法,如下所示:
CollectionTypeExtensions.HasItems(Validate.Argument("argument", argument));
我明白了:
匹配的最佳重载方法
'Validation.CollectionTypeExtensions.HasItems<int>(Validation.Argument<System.Collections.Generic.IEnumerable<int>>)'
有一些无效参数
根据我的研究,我需要这样做的东西称为“方差”,适用于接口和委托,但不适用于类(即:所有类都是不变的。)
也就是说,它可能会以另一种方式工作。想到的一个是将其重写为直接进入 T,如下所示:
public static Argument<T> HasItems<T, TElement>(this Argument<T> argument)
where T : IEnumerable<TElement>
{
if (!argument.Value.Any())
throw Error.Generic(argument.Name, "Collection contains no items.");
return argument;
}
..但这也不起作用,因为它需要在调用方法时显式指定 TElement 。我也可以回退到在类型约束中使用非通用 IEnumerable 接口,但是我必须找到一种方法将 IEnumerable 强制转换为 IEnumerable (这需要知道 T 在该上下文中是什么),或者复制Any() 的功能是为了测试是否存在任何项目,还有另一种扩展方法 (All) 会非常非常混乱,所以我宁愿避免它。
所以最终,我想我的问题是:如何让我的扩展方法正确附加?