代码之家  ›  专栏  ›  技术社区  ›  Danvil

如何同时遍历两个IEnumerable?

  •  51
  • Danvil  · 技术社区  · 14 年前

    我有两个可列举的: IEnumerable<A> list1 IEnumerable<B> list2 . 我想同时遍历它们,比如:

    foreach((a, b) in (list1, list2))
    {
        // use a and b
    }
    

    如果它们不包含相同数量的元素,则应引发异常。

    最好的方法是什么?

    8 回复  |  直到 6 年前
        1
  •  30
  •   Lasse V. Karlsen    14 年前

    以下是此操作的实现,通常称为zip:

    using System;
    using System.Collections.Generic;
    
    namespace SO2721939
    {
        public sealed class ZipEntry<T1, T2>
        {
            public ZipEntry(int index, T1 value1, T2 value2)
            {
                Index = index;
                Value1 = value1;
                Value2 = value2;
            }
    
            public int Index { get; private set; }
            public T1 Value1 { get; private set; }
            public T2 Value2 { get; private set; }
        }
    
        public static class EnumerableExtensions
        {
            public static IEnumerable<ZipEntry<T1, T2>> Zip<T1, T2>(
                this IEnumerable<T1> collection1, IEnumerable<T2> collection2)
            {
                if (collection1 == null)
                    throw new ArgumentNullException("collection1");
                if (collection2 == null)
                    throw new ArgumentNullException("collection2");
    
                int index = 0;
                using (IEnumerator<T1> enumerator1 = collection1.GetEnumerator())
                using (IEnumerator<T2> enumerator2 = collection2.GetEnumerator())
                {
                    while (enumerator1.MoveNext() && enumerator2.MoveNext())
                    {
                        yield return new ZipEntry<T1, T2>(
                            index, enumerator1.Current, enumerator2.Current);
                        index++;
                    }
                }
            }
        }
    
        class Program
        {
            static void Main(string[] args)
            {
                int[] numbers = new[] { 1, 2, 3, 4, 5 };
                string[] names = new[] { "Bob", "Alice", "Mark", "John", "Mary" };
    
                foreach (var entry in numbers.Zip(names))
                {
                    Console.Out.WriteLine(entry.Index + ": "
                        + entry.Value1 + "-" + entry.Value2);
                }
            }
        }
    }
    

    若要使其在仅一个序列的值用完时引发异常,请更改while循环,以便:

    while (true)
    {
        bool hasNext1 = enumerator1.MoveNext();
        bool hasNext2 = enumerator2.MoveNext();
        if (hasNext1 != hasNext2)
            throw new InvalidOperationException("One of the collections ran " +
                "out of values before the other");
        if (!hasNext1)
            break;
    
        yield return new ZipEntry<T1, T2>(
            index, enumerator1.Current, enumerator2.Current);
        index++;
    }
    
        2
  •  49
  •   Jon Skeet    7 年前

    你想要像 Zip LINQ运算符-.NET4中的版本总是在两个序列完成时截断。

    这个 MoreLINQ implementation 有一个 EquiZip 将抛出 InvalidOperationException 相反。

    var zipped = list1.EquiZip(list2, (a, b) => new { a, b });
    
    foreach (var element in zipped)
    {
        // use element.a and element.b
    }
    
        3
  •  17
  •   Samuel    6 年前

    简而言之,语言并没有提供一个干净的方法来做到这一点。枚举设计为一次对一个可枚举项执行。你可以很容易地模仿foreach为你做的事情:

    using(IEnumerator<A> list1enum = list1.GetEnumerator())
    using(IEnumerator<B> list2enum = list2.GetEnumerator())    
    while(list1enum.MoveNext() && list2enum.MoveNext()) {
            // list1enum.Current and list2enum.Current point to each current item
        }
    

    如果它们有不同的长度,该怎么办取决于你。也许在while循环完成后找出哪个元素仍然有元素,并继续使用那个元素,如果它们的长度应该相同,则抛出异常,等等。

        4
  •  3
  •   Anthony Pegram    14 年前

    在.net 4中,可以在 IEnumerable<T>

    IEnumerable<int> list1 = Enumerable.Range(0, 100);
    IEnumerable<int> list2 = Enumerable.Range(100, 100);
    
    foreach (var item in list1.Zip(list2, (a, b) => new { a, b }))
    {
        // use item.a and item.b
    }
    

    不过,它不会造成长度不等。不过,你可以随时测试。

        5
  •  3
  •   jpabluz    14 年前

    使用IEnumerable.getEnumerator,这样就可以在可枚举的范围内移动。注意,这可能有一些非常恶劣的行为,你必须小心。如果您想让它正常工作,那么就这样做;如果您想拥有可维护的代码,那么就使用两个foreach。

    如果要在代码中多次使用包装类,可以创建包装类或使用库(正如jon skeet建议的那样)以更通用的方式处理此功能。

    我建议的代码是:

    var firstEnum = aIEnumerable.GetEnumerator();
    var secondEnum = bIEnumerable.GetEnumerator();
    
    var firstEnumMoreItems = firstEnum.MoveNext();
    var secondEnumMoreItems = secondEnum.MoveNext();    
    
    while (firstEnumMoreItems && secondEnumMoreItems)
    {
          // Do whatever.  
          firstEnumMoreItems = firstEnum.MoveNext();
          secondEnumMoreItems = secondEnum.MoveNext();   
    }
    
    if (firstEnumMoreItems || secondEnumMoreItems)
    {
         Throw new Exception("One Enum is bigger");
    }
    
    // IEnumerator does not have a Dispose method, but IEnumerator<T> has.
    if (firstEnum is IDisposable) { ((IDisposable)firstEnum).Dispose(); }
    if (secondEnum is IDisposable) { ((IDisposable)secondEnum).Dispose(); }
    
        6
  •  2
  •   Jeffrey L Whitledge    7 年前
    using(var enum1 = list1.GetEnumerator())
    using(var enum2 = list2.GetEnumerator())
    {
        while(true)
        {
            bool moveNext1 = enum1.MoveNext();
            bool moveNext2 = enum2.MoveNext();
            if (moveNext1 != moveNext2)
                throw new InvalidOperationException();
            if (!moveNext1)
                break;
            var a = enum1.Current;
            var b = enum2.Current;
            // use a and b
        }
    }
    
        7
  •  1
  •   Jason Webb    14 年前

    你可以这样做。

    IEnumerator enuma = a.GetEnumerator();
    IEnumerator enumb = b.GetEnumerator();
    while (enuma.MoveNext() && enumb.MoveNext())
    {
        string vala = enuma.Current as string;
        string valb = enumb.Current as string;
    }
    

    C没有任何前科可以做你想做的事(我知道)。

        8
  •  1
  •   MartinStettner    14 年前

    使用 Zip 功能类似

    foreach (var entry in list1.Zip(list2, (a,b)=>new {First=a, Second=b}) {
        // use entry.First und entry.Second
    }
    

    不过,这不会引发异常……