【问题标题】:Conditionally remove elements from a List in Java 8 [duplicate]有条件地从 Java 8 中的列表中删除元素 [重复]
【发布时间】:2018-08-08 00:54:44
【问题描述】:

据我所知,Java 8 引入了一种可用于 Collection 类型的新方法:removeif()。它接受一个谓词,该谓词定义应删除元素的条件。它返回一个布尔值,其中真正的响应意味着至少一个项目已被删除,否则为 false: 我有这门课:

HotelPriceSummary {
     Hotel hotel;
     float price
}

List<HotelPriceSummary> allHotels;

Iterable<Hotel> discardedHotels

我想做类似的事情(显然existsIn是一个不存在的函数,是表达我想做的但是我没有找到方法)

allHotels.removeIf(h -> h.getHotel().existsIn (discardedHotels))

【问题讨论】:

  • 您可以从discardedHotels Iterable 生成HashSet(假设Hotel 覆盖equals 和hashCode)并使用set.contains()
  • 你能用Collection代替Iterable吗?它会让这变得简单得多。
  • 使用ArrayList.removeAll(Collection<?> c)
  • discardedHotels.forEach(h -> allHotels.removeIf(hps -> hps.getHotel().equals(h))) 怎么样?效率不高,我知道。
  • @Harshit 没有ArrayList,没有Collection,它们甚至不是同一个元素类型。

标签: java lambda collections java-8


【解决方案1】:

您无法有效地定位 Iterable 中的项目。我建议将它复制到一个临时集合(除非它已经是一个集合),然后调用contains()

Set<Hotel> discardedSet = new HashSet<>();
discardedHotels.forEach(discardedSet::add);
allHotels.removeIf(h -> discardedSet.contains(h.getHotel()));

如果您不介意 O(n*m) 复杂度,可以在 Iterable.forEach() 内调用 List.removeIf()

discardedHotels.forEach(h -> allHotels.removeIf(hps -> hps.getHotel().equals(h)))

【讨论】:

  • 集创建可以通过Set&lt;Hotel&gt; discardedSet = StreamSupport.stream(discardedHotels.spliterator(), false).collect(Collectors.toSet());简化
  • @Flown 感谢您的建议,但我不认为这是一种简化。
猜你喜欢
  • 1970-01-01
  • 2017-12-15
  • 2018-08-12
  • 2014-05-12
  • 2020-07-05
  • 2014-05-06
  • 1970-01-01
  • 2020-05-27
  • 2021-03-07
相关资源
最近更新 更多