使用Linq隐式演员

时间:2015-12-01 14:23:41

标签: c# .net extension-methods linq-to-objects

假设我有一个带整数的列表(var identifiers = Enumerable.Empty<int>())。

使用此列表,我可以将各个项目转换为另一种类型:

var castedIdentifiersLong = identifiers.Cast<long>();
var castedIdentifiersString = identifiers.Cast<string>();

我们可以使用Select<TSource, TTarget>()来使用隐式转换:

var mappedIdentifiersLong = identifiers.Select<int, long>(x => x);
var mappedIdentifiersString = identifiers.Select<int, string>(x => x);

显然最后一个语句失败了,因为int无法隐式转换为string。这是故意的。

有没有办法可以定义一个扩展方法(比如CastImplicitly<T>),我只能定义两个泛型类型中的一个,它会从源枚举中找出第一个类型?

var unwanted = identifiers.ImplicitCast<string>();
var wanted = identifiers.ImplicitCast<long>();

在这种情况下,unwanted甚至不应该编译,因为int不能隐式转换为string。但另一方面,wanted应该编译,因为它可以隐式地转换为long

1 个答案:

答案 0 :(得分:3)

  

有没有办法可以定义一个扩展方法(比如CastImplicitly<T>),我只能定义两个泛型类型中的一个,它会从源枚举中找出第一个类型?

如果输入是通用的,则不会。编译器无法部分推断泛型参数 - 您必须指定所有通用参数或无指定编译器。

即使可以,编译器仍然不允许在泛型类型之间进行隐式转换。

如果您希望编译器在编译时识别无效的强制转换,您可以进行显式转换:

var mappedIdentifiersString = identifiers.Select(x => (string)x);  // fails at compile time if x is an int.

它可能无法在编译时捕获每个可能的无效转换(例如,在编译时始终允许转换为来自object的转换)但是它对于您的特定{{ 1}}到int方案。