在java8中积累对的函数方法

时间:2015-06-12 16:36:44

标签: java functional-programming java-8 java-stream

这是我试图转化为函数式编程代码的一些命令式代码:

public class Person {
    String name;
    Token token;

    public Person(String name, Token token) {
        this.name = name;
        this.token = token;
    }
}
public class Token {
    String id;
    boolean isValid;
    public Token(String id, boolean isValid) {
        this.id = id;
        this.isValid = isValid;
    }
    public String getId() { return id; }
    public boolean isValid() {return isValid;}
}
public static List<Token> getTokensForPerson(String name) {...}


public static List<Person> getPeople1 (String[] names) {

    List<Person> people = new ArrayList<Person> ();
    for (String name: names) {
        List<Token> tokens = getTokensForPerson(name);
        for (Token token: tokens) {
            if (token.isValid()) {
                people.add(new Person(name, token));
            }
        }

    }
    return people;
}

这是我尝试以功能方式完成的。

public static List<Person> getPeople2 (String[] names) {

    return Arrays.stream(names).map(name -> getTokensForPerson(name))
        .flatMap(tokens -> tokens.stream().filter(token -> token.isValid))
        .map(token -> new Person(name, token))   // <== compiler error here. "Cannot resolve symbol 'name'"
        .collect(Collectors.toList());
}

但是它没有编译,因为在上一个map操作中我需要引用name来创建Person对象,name当时不可用。有什么想法吗?

2 个答案:

答案 0 :(得分:2)

您可以移动map中的flatMap步骤:

return Arrays.stream(names)
        .<Person>flatMap(
                name -> getTokensForPerson(name).stream()
                        .filter(Token::isValid)
                        .map(token -> new Person(name, token)))
        .collect(Collectors.toList());

这样您也可以访问name变量。

StreamEx - 基于解决方案的时间较短,但它需要第三方库:

return StreamEx.of(names)
               .cross(name -> getTokensForPerson(name).stream())
               // Here we have the stream of entries 
               // where keys are names and values are tokens
               .filterValues(Token::isValid)
               .mapKeyValue(Person::new)
               .toList();

答案 1 :(得分:0)

是否可以创建TokenExtended类扩展令牌,并添加名称,并从getTokensForPerson而不是List<TokenExtended>返回List<Token>

public class TokenExtended extends Token {
    private String name;
    public TokenExtended(String name, String id, boolean isValid) {
        super(id, isValid);
        this.name = name;
    }
}

这样你的代码就可以了

    Arrays.stream(names).map(name -> getTokensForPerson(name)).flatMap(tokens -> tokens.stream().filter(token -> token.isValid))
            .map(token -> new Person(token.name, token)).collect(Collectors.toList());