LINQ:根据重量/尺寸将列表分成组
本文关键字:列表 LINQ | 更新日期: 2023-09-27 18:14:07
我有很多使用LINQ的例子,如何根据每个列表中的最大条目将列表划分为子列表。但是在这种情况下,我对使用sizemb作为权重的子列表感兴趣-每个列表的最大总文件大小为9mb。
public class doc
{
public string file;
public int sizemb;
}
var list = new List<doc>()
{
new doc { file = "dok1", sizemb = 5 },
new doc { file = "dok2", sizemb = 5 },
new doc { file = "dok3", sizemb = 5 },
new doc { file = "dok4", sizemb = 4 },
};
int maxTotalFileSize = 9;
然后将上述列表分为3个列表。如果任何"文件"大于9mb,它们应该在自己的列表中。
我在这里做了一个非linq版本:
var lists = new List<List<doc>>();
foreach (var item in list)
{
//Try and place the document into a sub-list
var availableSlot = lists.FirstOrDefault(p => (p.Sum(x => x.sizemb) + item.sizemb) < maxGroupSize);
if (availableSlot == null)
lists.Add(new List<doc>() { item });
else
availableSlot.Add(item);
}
您可以使用以下方法:
IEnumerable<IList<doc>> SplitDocumentList(IEnumerable<doc> allDocuments, int maxMB)
{
var lists = new List<IList<doc>>();
var list = new List<doc>();
foreach (doc document in allDocuments)
{
int totalMB = list.Sum(d => d.sizemb) + document.sizemb;
if (totalMB > maxMB)
{
lists.Add(list);
list = new List<doc>();
}
list.Add(document);
}
if (list.Count > 0)
lists.Add(list);
return lists;
}
下面是一个演示:http://ideone.com/OkXw7C
dok1
dok2
dok3,dok4
您可以使用Aggregate函数来完成此操作,组by仅在比较值时才工作,而不是基于何时开始新组的任意条件
list.Aggregate(new List<List<doc>>(), (acc,d) => {
if(acc.last().Sum(x => x.sizemb) + d.sizemb > 9) {
acc.Add(new List<doc>());
}
acc.last().Add(d);
return acc;
}
)