Как разделить список на основе индекса в C #? - PullRequest
10 голосов
/ 08 апреля 2011

На данный момент у меня есть список, содержащий около 190 элементов.Как я могу разделить список на более мелкие списки, содержащие не более 50 элементов в каждом списке?

Результатом могут быть списки из 50, 50, 50 и 40 элементов.

Ответы [ 4 ]

21 голосов
/ 08 апреля 2011

Если вы имеете в виду List<T>, вы можете использовать метод GetRange несколько раз. Черт возьми, вы можете сделать это с помощью LINQ:

var lists = Enumerable.Range(0, (list.Count + size - 1) / size)
      .Select(index => list.GetRange(index * size,
                                     Math.Min(size, list.Count - index * size)))
      .ToList();

Или вы можете просто использовать цикл, конечно:

public static List<List<T>> Split(List<T> source, int size)
{
    // TODO: Prepopulate with the right capacity
    List<List<T>> ret = new List<List<T>>();
    for (int i = 0; i < source.Count; i += size)
    {
        ret.Add(source.GetRange(i, Math.Min(size, source.Count - i)));
    }
    return ret;
}

Это несколько более эффективно, чем использование GroupBy, хотя оно ограничено List<T> в качестве входа.

У нас есть другая реализация, использующая IEnumerable<T> в MoreLINQ в Batch.cs .

10 голосов
/ 08 апреля 2011

Вы можете использовать LINQ:

var list = Enumerable.Range(1, 190);
var sublists = list
    .Select((x, i) => new { Index = i, Value = x })
    .GroupBy(x => x.Index / 50)
    .Select(x => x.Select(v => v.Value).ToList())
    .ToArray();
1 голос
/ 08 апреля 2011

Я попытался рекурсивный подход.Просто чтобы посмотреть, как это будет выглядеть.

List<List<T>> SplitIntoChunks<T>(IEnumerable<T> originalList, int chunkSize)
{
    if(originalList.Take(1).Count() == 0)
    {
        return new List<List<T>>();
    }

    var chunks = new List<List<T>> {originalList.Take(chunkSize).ToList()};
    chunks.AddRange(SplitIntoChunks(originalList.Skip(chunkSize), chunkSize));
    return chunks;
}
0 голосов
/ 08 апреля 2011
var list = new List<int>(Enumerable.Range(1,190));
var page_size = 50;
var max_pages = 1 + list.Count() / page_size;

for(int page = 1; page <= max_pages; page++) {
  var chunk = list.Skip(page_size * (page-1)).Take(page_size);
  // do whatever
}
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...