Есть ли в .Net метод, похожий на почтовый индекс? - PullRequest
12 голосов
/ 19 сентября 2008

В Python есть действительно аккуратная функция под названием zip, которую можно использовать для перебора двух списков одновременно:

list1 = [1, 2, 3]
list2 = ["a", "b", "c"]
for v1, v2 in zip(list1, list2):
    print v1 + " " + v2

Приведенный выше код должен выдать следующее:

1 a
2 b
3 c

Интересно, есть ли такой метод, доступный в .Net? Я думаю написать сам, но нет смысла, если он уже доступен.

Ответы [ 4 ]

25 голосов
/ 19 сентября 2008

Обновление: встроено в C # 4 как System.Linq.Enumerable.Zip Method

Вот версия C # 3:

IEnumerable<TResult> Zip<TResult,T1,T2>
    (IEnumerable<T1> a,
     IEnumerable<T2> b,
     Func<T1,T2,TResult> combine)
{
    using (var f = a.GetEnumerator())
    using (var s = b.GetEnumerator())
    {
        while (f.MoveNext() && s.MoveNext())
            yield return combine(f.Current, s.Current);
    }
}

Отбросил версию C # 2, поскольку она показывала свой возраст.

8 голосов
/ 19 сентября 2008

Насколько я знаю, нет. Я написал одно для себя (а также несколько других полезных расширений и поместил их в проект под названием NExtension на Codeplex.

Очевидно, что параллельные расширения для .NET имеют функцию Zip.

Вот упрощенная версия от NExtension (но, пожалуйста, проверьте ее для более полезных методов расширения):

public static IEnumerable<TResult> Zip<T1, T2, TResult>(this IEnumerable<T1> source1, IEnumerable<T2> source2, Func<T1, T2, TResult> combine)
{
    using (IEnumerator<T1> data1 = source1.GetEnumerator())
    using (IEnumerator<T2> data2 = source2.GetEnumerator())
        while (data1.MoveNext() && data2.MoveNext())
        {
            yield return combine(data1.Current, data2.Current);
        }
}

Использование:

int[] list1 = new int[] {1, 2, 3};
string[] list2 = new string[] {"a", "b", "c"};

foreach (var result in list1.Zip(list2, (i, s) => i.ToString() + " " + s))
    Console.WriteLine(result);
5 голосов
/ 19 сентября 2008

Нет, в .NET такой функции нет. Вы выкатили свой собственный. Обратите внимание, что C # не поддерживает кортежи, поэтому отсутствует синтаксис типа Python.

Вы можете использовать что-то вроде этого:

class Pair<T1, T2>
{
    public T1 First { get; set;}
    public T2 Second { get; set;}
}

static IEnumerable<Pair<T1, T2>> Zip<T1, T2>(IEnumerable<T1> first, IEnumerable<T2> second)
{
    if (first.Count() != second.Count())
        throw new ArgumentException("Blah blah");

    using (IEnumerator<T1> e1 = first.GetEnumerator())
    using (IEnumerator<T2> e2 = second.GetEnumerator())
    {
        while (e1.MoveNext() && e2.MoveNext())
        {
            yield return new Pair<T1, T2>() {First = e1.Current, Second = e2.Current};
        }
    }
}

...

var ints = new int[] {1, 2, 3};
var strings = new string[] {"A", "B", "C"};

foreach (var pair in Zip(ints, strings))
{
    Console.WriteLine(pair.First + ":" + pair.Second);
}
2 голосов
/ 19 сентября 2008

Есть также один в F #:

let zipped = Seq.zip firstEnumeration secondEnumation

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...