如何从C#中的两个列表创建单个对象列表?

时间:2011-08-18 16:21:52

标签: c# list object

我有两个对象列表。列表A和列表B.我需要创建列表C,它将列表A和列表B组合成对。例如:

List A
object a1
object a2
object a3

List B
object b1
object b2
object b3

List C (creates pairs)
object c1 (object a1, object b1)
object c2 (object a2, object b2)
object c3 (object a3, object b3)

6 个答案:

答案 0 :(得分:37)

您可以在System.Linq中使用Enumerable.Zip()方法。

IEnumerable<Tuple<A, B>> pairs = listA.Zip(listB, (a, b) => Tuple.Create(a, b));

使用此结果可枚举的示例:

foreach (Tuple<A, B> pair in pairs)
{
    A a = pair.Item1;
    B b = pair.Item2;
}

遗憾的是,没有一个重载可以自动化.NET中的元组。这样的扩展看起来像这样:

public static IEnumerable<Tuple<TFirst, TSecond>> Zip<TFirst, TSecond>(this IEnumerable<TFirst> first, IEnumerable<TSecond> second)
{
    return first.Zip(second, Tuple.Create);
}

这意味着您可以编写看起来像的代码:

IEnumerable<Tuple<A, B>> pairs = listA.Zip(listB);

注意:另一种选择是创建匿名类型而不是Tuple,但这种方法的缺点是你将无法(有用地)将生成的IEnumerable传递给它所创建的方法到没有名字的类型。

答案 1 :(得分:5)

这样就可以了:

public static IEnumerable<Tuple<T, U>> CombineWith<T, U>(this IEnumerable<T> first, IEnumerable<U> second)
{
    using (var firstEnumerator = first.GetEnumerator())
    using (var secondEnumerator = second.GetEnumerator())
    {
        bool hasFirst = true;
        bool hasSecond = true;

        while (
            // Only call MoveNext if it didn't fail last time.
            (hasFirst && (hasFirst = firstEnumerator.MoveNext()))
            | // WARNING: Do NOT change to ||.
            (hasSecond && (hasSecond = secondEnumerator.MoveNext()))
            )
        {
            yield return Tuple.Create(
                    hasFirst ? firstEnumerator.Current : default(T),
                    hasSecond ? secondEnumerator.Current : default(U)
                );
        }
    }
}

编辑:我非常喜欢Paul's回答。

答案 2 :(得分:3)

这样的事情:

 var listA = new object[] { "1", "2", "3" };
 var listB = new object[] { "a", "b", "c" };
 var listC = Enumerable.Zip(listA,listB, (x,y)=>new {x,y});

 foreach (var item in listC)
 {
    Console.WriteLine("{0},{1}", item.x,item.y);
 }

输出:    1,一个    2,B    3,C

答案 3 :(得分:3)

.NET Core 3具有new overload for the Zip method。它需要两个IEnumerable,并创建一个IEnumerable,其中包含来自两个输入IEnumerable的元素的值元组。

IEnumerable<(A a, B b)> pairs = listA.Zip(listB);

您可以通过多种方式使用结果:

foreach((A a, B b) in pairs)
{
    // Do something with a and b
}
foreach(var (a, b) in pairs)
{
    // Do something with a and b
}
foreach(var pair in pairs)
{
    A a = pair.a;
    B b = pair.b;
}

答案 4 :(得分:1)

答案 5 :(得分:0)

使用System.Linq Enumerable.Zip方法(不指定函数参数Func)的潜在解决方案如下(请注意,我已经使用int来提供一个简单的具体示例): / p>

List<int> list1 = new List<int>() {1,2,3,4,5};
List<int> list2 = new List<int>() {5,4,3,2,1};

IEnumerable<(int, int)> pairs;
pairs = list1.Zip(list2);
Console.WriteLine(string.Join(", ", pairs));

输出:

(1, 5), (2, 4), (3, 3), (4, 2), (5, 1)