c# - 在编译时不知道类型的情况下将 IEnumerable<A> 转换为 IEnumerable<B>

标签 c# ienumerable

我有两种类型:我们称它们为 A 和 B。可以使用适配器方法将 A 转换为 B。

然后我在 List<A> 中得到了 A 的集合。 (它可以是任何支持 IEnumerable<A> 的集合类型)。

我现在想从 IEnumerable<A> 进行转换至IEnumerable<B> 。我知道Type A 和 B 中的每一个,并且我有一个将 A 转换为 B 的方法,但我的方法和/或类本身没有模板化,因此我无权访问模板类型;例如IEnumerable<T> 中的 T .

我实际上想编写这个 ConvertCollection 方法,其中我知道“from”的类型为 IEnumerable<{something}> :

object ConvertCollection(object from, Type fromType, Type toType, Converter converter);

我的转换器如下所示:

delegate object Converter(object from);

我的尝试让我留在这里:

object ConvertCollection(object from, Type fromType, Type toType, Converter converter)
{
    return ((IEnumerable<object>)from).Select(converter);
}

这部分有效。如果我这样调用它

ConvertCollection(new List<A>() { new A() }, typeof(A), typeof(B), AToBConverter);

返回的集合确实包含 B 的集合,但该集合本身的类型为 IEnumerable<object> ,不是IEnumerable<B> ,因为我不知道如何转换到 IEnumerable<{toType}> 。 (这很重要,因为结果需要序列化)。

我可以从另一端攻击它并创建正确的返回类型,如下所示:

var result = Activator.CreateInstance(typeof(List<>).MakeGenericType(toType));
// TODO: populate result here
return result;

但问题是,要实现 TODO 部分,我需要对结果调用 List<> 方法,但我无法将其转换为 List<> 的任何类型由于 Co/ContraVariance 规则,所以即使我知道该类型支持 List<>方法,我无法使用它们来填充列表;例如使用 Add()。

有没有一种方法可以在不使用“动态”且没有太多反射的情况下做到这一点?我知道我可以通过反射找到并调用 Add() 方法,但似乎没有必要。

.NET 4.0 顺便说一句

-- 澄清

正如 Euphoric 正确推测的那样,我在上面尝试过但严重失败,我在运行时知道类型 A 和 B,但在编译时不知道它们。因此,直接使用泛型不是一种选择。我确实知道集合(提供的和必须返回的)实现了通用 IEnumerable<> 。这一切都是固定的,超出了我的控制范围。 (我已经相应地调整了标题)。

** 编辑 2:修复了一些导致 <> 的格式不显示(很容易不小心省略反引号!)

最佳答案

使用 LINQ Select 方法:

var result = listA.Select(a => Converter(a));

由于您使用的是 .NET 4.0,因此您确实应该避免使用对象并使用泛型。

关于c# - 在编译时不知道类型的情况下将 IEnumerable<A> 转换为 IEnumerable<B>,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/17598219/

相关文章:

C# 合约设计 : How to Guarantee method and function

c# - 当用户是管理员时包括 IsDeleted 值

c# - SWIG 中的委托(delegate) - C#

c# - Linq 按属性排序,然后按原始顺序

c# - C# mvc 中的 IEnumerable<SelectListItem>

c# - 我如何请求提升注册管理机构对 HKLM 的访问权限?

c# - 在 WinForms 应用程序中将文本居中放置在按钮上

c# - 用一行代码填充 Enumerable.ToList

c# - 为什么我需要在程序逻辑中枚举我的列表才能在这种情况下工作?

c# - 将数组对转换为数组对(如果可能,使用 LINQ)