猿问

C# / .NET 比较两个大列表并从两个列表中查找缺失的项目

所以基本上我有两个大列表,如下所示:


public class Items

{

 public string ItemID { get; set; }

}



var oldList = new List<Items>(); // oldList


var newList = new List<Items>(); // new list

两个列表都非常大,如果它们都很大(超过 30 秒),由于执行时间很短,因此简单的双 foreach 是不够的。


在我在 stackoverflow 上提出的上一个问题中,我得到了关于如何比较这两个相同列表并找出哪些项目具有不同 QuantitySold 参数的回复,然后将其存储在名为“DifferentQuantityItems”的第三个列表中,如下所示:


var differentQuantityItems =

    (from newItem in newList

     join oldItem in oldList on newItem.ItemID equals oldItem.ItemID

     where newItem.QuantitySold != oldItem.QuantitySold

     select newItem).ToList();

现在我想从这两个列表中得到以下信息:


- A list of items that are present in newList, but not in oldList


- A list of items that are present in oldList, but not in newList

我怎样才能做到这一点?有人可以帮我吗?


PS我会“知道”其中一个列表中缺少任一项目的方式是通过属性“ItemID”......


一只名叫tom的猫
浏览 129回答 4
4回答

米脂

已编辑除了会工作得更快。在这里您可以了解它的性能var missedOld = oldList.Except(newList, new ItemsEqualityComparer());var oldList= oldList.Except(missedOld, new ItemsEqualityComparer());旧答案缺少项目的两个不同列表var missedOld = oldList.Where(x => !newList.Select(i => i.ItemID).Contains(x.ItemID))&nbsp;var missedNew = newList.Where(x => !oldList.Select(i => i.ItemID).Contains(x.ItemID))一个列表中的所有错过的项目:oldList.Concat(newList).GroupBy(x => x.ItemID).Where(x => x.Count() < 2).Select(x => x.Value).ToList()

守着星空守着你

您是否考虑过将列表转换为哈希集并使用 except 方法?请参阅两个列表之间的差异并且:有没有办法在c#中获得两组对象之间的差异

Qyouu

var&nbsp;items&nbsp;=&nbsp;new&nbsp;List<int>(oldList.Select(x&nbsp;=>&nbsp;x.ItemID&nbsp;)); var&nbsp;missingValues&nbsp;=&nbsp;newList.Where(x&nbsp;=>&nbsp;!diffids.Contains(x.ItemID)).ToList();您也可以使用除。

九州编程

如果列表足够大以至于嵌套循环需要 30 秒,我建议您将每个列表的项目放入相应的 HashSet 并使用它来查找异常。哈希表将以 O(1) 或 O(log N) 缩放,而比较 2 个未排序的列表是 O(n^2)。也就是说,尝试使用 Linq except()var&nbsp;notinNewList&nbsp;=&nbsp;oldList.Except(newList);如果我没记错的话,.Except() 的内部实现依赖于 HashSets其次,如果列表已排序,或者可以预先排序,那么您可以在没有嵌套循环的情况下一次性进行线性传递,这可能比任何方法都快。我不建议使用 List.Contains() 因为它是一个线性实现,这将导致您试图避免的 O(n^2) 相同,尽管由于 Linq 语法糖它看起来更漂亮。
随时随地看视频慕课网APP
我要回答