我有两种类型:我们称它们为 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/