6

我正在尝试理解 Java 8 中的 Lambda。

假设我有一个 Person 类,如下所示:

public class Person implements {
    String name;
    GenderEnum gender;
    int age;
    List<Person> children;
}

现在我想做的是找到所有女性,有小于 10 岁的孩子的人。

Pre java 8 我会这样做:

List<Person> allPersons = somePeople();
List<Person> allFemaleWithChildren = new ArrayList<>();
for(Person p : allPersons) {
    for(Person child : p.getChildren()) {
        if(child.getAge() < 10 && p.getGender() == GenderEnum.Female) {
            allFemaleWithChildren.add(p);
        }
    }
}

现在 allFemaleWithChildren 应该有我想要的。我一直在尝试使用流做同样的事情我认为我需要使用某种映射、过滤和减少

allPersons.stream()
//filter females
.filter(p -> p.getGender == GenderEnum.Female)
//get the children
.map(c -> c.getChildren())
//filter the ones that are less than 10 years
.filter(c -> c.getAge() < 10)
//return a list with the result
.collect(Collectors.toList())

但是这段代码无法编译。我错过了什么。

另外,我不明白 reduce 方法可以用来做什么。

编译器说 cannot resolve method getAge()。这是因为c显然是一个集合而不是集合中的项目,这正是我想要的。

4

2 回答 2

9

目前(一旦你修复了编译错误)你将返回一个子列表。假设在您的原始代码中,您打算在找到 10 岁以下的孩子后立即中断,等效代码可能如下所示:

allPersons.stream()
    //filter females
    .filter(p -> p.getGender() == GenderEnum.Female)
    //only keep females with at least one child < 10
    .filter(f -> f.getChildren().stream()
                    .anyMatch(c -> c.getAge() < 10))
    //return a list with the result
    .collect(Collectors.toList())

事实上,正如下面评论的那样,您可以使用一些静态导入,添加辅助方法并重构原始代码以使其更具可读性:

allPersons.stream()
    .filter(this::female)
    .filter(this::hasChildrenUnder10)
    .collect(toList())

//...

private boolean female(Person p) { return p.getGender() == Female; }
private boolean hasChildrenUnder10(Person parent) {
    return parent.getChildren().stream()
                    .anyMatch(c -> c.getAge() < 10));
}
于 2014-03-26T13:45:04.387 回答
1

您有 2 个for循环,这意味着在某些时候您需要另一个流。在这里,当您调用 时map,您会将您的母亲映射到孩子的列表中。然后你继续,好像你有一个孩子的流,但实际上你有一个孩子的集合。

于 2014-03-26T13:49:12.083 回答