[英]Java8 Stream with Multiple List of Objects
我有两个列表,我想根据 Object1 和 Object2 的属性在其中执行一些操作。
对象 1:
public class Object1 {
private String prop1;
private String prop2;
}
对象 2:
public class Object2 {
private String prop1;
private String prop2;
private String prop3;
private boolean prop4;
}
我想使用流而不是使用大方法。
假设我们有以下列表:
List<Object1> object1List = // List of Object1's
List<Object2> object2List = // List of Object2's
现在,我想比较两个列表中的 prop1 和 prop2,如果它们相等,我想从 Object2 获取 prop3 并基于 prop3 现在我想从 Object2 中查看 prop4 为 true 的所有元素,并确保所有元素都为 true 。
例子:
Object1:
prop1 | prop2
test1 value1
test1 value2
Object2:
prop1 | prop2 | prop3 | prop4
test1 value1 some true
test1 value2 some1 true
test2 value3 some true
现在从上面的示例中,我想同时流式传输列表并比较列表,如果 prop1 和 prop2 相等,我想在“some”的情况下获得 prop3 并使用 prop3 值获得所有 prop4 值“true” some" 并查看 prop1 和 prop2 是否在 Object 的第一个列表中并将它们收集在一个集合中。 我可以用传统的方式做到这一点,但流看起来更易读,代码行更少。 我只是在学习流,任何想法都值得赞赏!
由于我很无聊,无法弄清楚这个问题在问什么,这里有一个使用 Streams 实现此功能的示例。 这里的关键是根据适用于流的每个元素的 Predicate 过滤/(任何)匹配。
主程序
import java.util.*; import java.util.stream.*;
class Main
{
public static void main(String[] args)
{
List<Object1> l1 = new ArrayList<>();//referred to as "first list" in comments
List<Object2> l2 = new ArrayList<>();//referred to as "second list" in comments
l1.add(new Object1("test1","value1"));
l1.add(new Object1("test1","value2"));
l2.add(new Object2("test1", "value1", "some", true));
l2.add(new Object2("test1", "value2", "some1", true));
l2.add(new Object2("test2", "value3", "some", true));
//This is the stream of objects2 in second list that matched both prop1 and prop2 for at least one object1 in first list
Stream<Object2> matchingProp1andProp2 = l2.stream().filter(obj2 -> l1.stream().anyMatch(obj1 -> obj1.prop1().equals(obj2.prop1()) && obj1.prop2().equals(obj2.prop2())));
matchingProp1andProp2.forEach(match -> {
//get the ones in second list that have same prop3 as the match and prop4 being true
Stream<Object2> matchingCriteria = l2.stream().filter(obj2 -> match.prop3().equals(obj2.prop3()) && obj2.prop4());
System.out.println(matchingCriteria.collect(Collectors.toList()));
});
}
}
选项 1 - 如果您认为单独阅读每个条件更容易:
Set<Object2> result = l2.stream().filter(o -> o.getProp3().equals("some"))
.filter(o -> l1.contains(new Object1(o.getProp1(), o.getProp2())))
.filter(Object2::isProp4)
.collect(Collectors.toSet());
选项 2 - 具有所有条件的一个过滤器:
result = l2.stream().filter(o -> o.getProp3().equals("some") && l1.contains(new Object1(o.getProp1(), o.getProp2())) && o.isProp4() ).collect(Collectors.toSet());
我假设您的 Object1 覆盖了 equals 和 hasCode,因此您可以使用 contains。
如果不是这种情况,则有点复杂,因为您必须分别比较 prop1 和 prop2,而不是对象本身:
result = l2.stream().filter(o -> o.getProp3().equals("some") && o.isProp4())
.filter(o1 -> l1.stream().anyMatch(o2 -> o2.getProp1().equals(o1.getProp1()) && o2.getProp2().equals(o1.getProp2())))
.collect(Collectors.toSet());
声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.