使用 Zip() 合并不同长度的数组而不丢失值

fir*_*mer 6 .net c# linq arrays

在下面的代码中,我合并了两个类型为int和的数组string。第一个的长度大于第二个,因此最后一个索引(即5)不会被合并:

int[] numbers = new[] { 1, 2, 3, 4, 5 };
string[] words = new string[] { "one", "two", "three", "four" };

var numbersAndWords = numbers.Zip(words, (n, w) => new { Number = n, Word = w });
foreach (var nw in numbersAndWords)
{
    Console.WriteLine(nw.Number + nw.Word);
}
Run Code Online (Sandbox Code Playgroud)

我想知道一种合并的方法。例如,null在存在的最后一个字符串之后创建一个或空字符串words,并使用它与最后一个索引合并numbers。无法弄清楚。

编辑: 我得到的结果

1one
2two
3three
4four
Run Code Online (Sandbox Code Playgroud)

我想要的结果

1one
2two
3three
4four
5
Run Code Online (Sandbox Code Playgroud)

谢谢!

编辑:不是重复的,我的另一个问题是关于在空对象上调用方法。

Mar*_*zek 5

您可以轻松编写自己的类似 LINQ 的扩展方法来执行此操作:

public static class MyEnumerable
{
    public static IEnumerable<TResult> ZipWithDefault<TFirst, TSecond, TResult>(this IEnumerable<TFirst> first, IEnumerable<TSecond> second, Func<TFirst, TSecond, TResult> selector)
    {
        bool firstMoveNext, secondMoveNext;

        using (var enum1 = first.GetEnumerator())
        using (var enum2 = second.GetEnumerator())
        {
            while ((firstMoveNext = enum1.MoveNext()) & (secondMoveNext = enum2.MoveNext()))
                yield return selector(enum1.Current, enum2.Current);

            if (firstMoveNext && !secondMoveNext)
            {
                yield return selector(enum1.Current, default(TSecond));
                while (enum1.MoveNext())
                {
                    yield return selector(enum1.Current, default(TSecond));
                }
            }
            else if (!firstMoveNext && secondMoveNext)
            {
                yield return selector(default(TFirst), enum2.Current);
                while (enum2.MoveNext())
                {
                    yield return selector(default(TFirst), enum2.Current);
                }
            }
        }
    }
}
Run Code Online (Sandbox Code Playgroud)

但如果您的源始终是一对数组,那么简单地使用for循环可能会更容易:

public static IEnumerable<TResult> ZipWithDefault<TFirst, TSecond, TResult>(this TFirst[] first, TSecond[] second, Func<TFirst, TSecond, TResult> selector)
{
    var maxLength = Math.Max(first.Length, second.Length);

    for(var i = 0; i < maxLength; i++)
    {
        var firstItem = i < first.Length ? first[i] : default(TFirst);
        var secondItem = i < second.Length ? second[i] : default(TSecond);
        yield return selector(firstItem, secondItem);

    }
}
Run Code Online (Sandbox Code Playgroud)