此问题与here
略有不同我有一组数字,我想要生成一个段列表。段的结尾包含在下一段的开始元素中。我希望每个段每个都有3个(在本例中)。这是一个例子:
var origArray = new[] {1,2,3,4,5,6};
我希望结果是:
{ {1,2,3}, {3,4,5}, {5,6} }
传统的for循环可以实现这一目标,但只是想知道是否有人以LINQ y 的方式做到了这一点。
答案 0 :(得分:2)
尝试添加Microsoft的Reactive Framework Team的Interactive Extensions - 只需NuGet“Ix-Main”。
然后你可以这样做:
var origArray = new[] {1,2,3,4,5,6};
var result = origArray.Buffer(3, 2);
这两个参数是“分组的数量”,3
和“要跳过的数量”,2
。
结果如您所愿:{ {1,2,3}, {3,4,5}, {5,6} }
。
这是https://github.com/Reactive-Extensions/Rx.NET/blob/master/Ix.NET/Source/System.Interactive/Buffer.cs的实施:
private static IEnumerable<IList<TSource>> Buffer_<TSource>(this IEnumerable<TSource> source, int count, int skip)
{
var buffers = new Queue<IList<TSource>>();
var i = 0;
foreach (var item in source)
{
if (i%skip == 0)
buffers.Enqueue(new List<TSource>(count));
foreach (var buffer in buffers)
buffer.Add(item);
if (buffers.Count > 0 && buffers.Peek()
.Count == count)
yield return buffers.Dequeue();
i++;
}
while (buffers.Count > 0)
yield return buffers.Dequeue();
}
答案 1 :(得分:0)
使用Linq执行此操作的两种可能方法。
var segmentedArray1 = origArray
.Select((item, index) => new { item, index })
.GroupBy(x => x.index/3)
.Select(@group => @group.Select(x=>x.item));
var segmentedArray2 = Enumerable.Range(0, origArray.Count())
.GroupBy(x => x/3)
.Select(@group => @group.Select(index => origArray[index]));
这里是dotnetfiddle