it-swarm.com.ru

Java 8 лямбда-предикат цепочки?

Я не могу заставить его скомпилировать, возможно ли даже связать лямбды-предикаты?

Arrays.asList("1","2","3").stream().filter( (e -> e=="1" ).or(e-> e=="2") ).count();

Или единственный способ - явно создать предикат, а затем объединить его так:

Predicate<String> isOne= e -> e=="1";
Arrays.asList("1","2","3").stream().filter( isOne.or(e -> e=="2") ).count();

Или есть более «функционально элегантный» способ добиться того же? 

31
user1606576

Вы можете использовать:

((Predicate<String>) e -> e.equals("1")).or(e -> e.equals("2"))

но это не очень элегантно Если вы указываете условия в строке, просто используйте одну лямбду:

e -> e.equals("1") || e.equals("2")
42
Chris Jester-Young

Я работал над аналогичной проблемой цепочки предикатов и придумал следующее

public static <T> Predicate<T> chain (Function<T,Predicate<T>> mapFn, T[]args) {
    return Arrays.asList(args)
        .stream()
        .map(x->mapFn.apply(x))
        .reduce(p->false, Predicate::or);
}

Первый параметр в цепочке - это лямбда (функция), которая возвращает лямбда (предикат), поэтому для этого требуется пара стрелок

public static void yourExample() {
    String[] filterVals = { "1", "2" };

    Arrays.asList("1","2","3")
        .stream()
        .filter(chain(x-> (y-> y.equals(x)), filterVals))
        .count();
}

Для сравнения вот чего я пытался добиться ...

public static void myExample() {
    String[] suffixes = { ".png", ".bmp" };
    Predicate<String> p = chain (x-> y-> y.endsWith(x), suffixes);
    File[] graphics = new File("D:/TEMP").listFiles((dir,name)->p.test(name));
    Arrays.asList(graphics).forEach(System.out::println);
}
10
Sarah Phillips

Вы можете использовать с магическим методом $:

<T> Predicate<T> $(Predicate<T> p) {
     return p;
}

Arrays.asList("1", "2", "3").stream().filter( $(e -> e=="1").or(e -> e=="2") ).count();
8
Vitaliy Oliynyk

Это всего лишь небольшое дополнение к ответу @Chris Jester-Young. Можно сократить выражение, используя ссылку на метод:

((Predicate<String>) "1"::equals).or("2"::equals)
4
Anton Balaniuc
Arrays.asList("1","2","3").stream().filter( Arrays.asList("1", "2")::contains).count();

и да, метод "либо" является хорошей идеей

    public static void main(String[] args) {
        long count = Arrays.asList("1","2","3").stream().filter(either("1"::equals).or("2"::equals)).count();
        System.out.println(count);
    }

    private static <T> Predicate<T> either(Predicate<T> predicate) {
        return predicate;
    }

или вы можете использовать import static Java.util.function.Predicate.isEqual; и написать isEqual("1").or(isEqual("2"))

4
Алексей Лумпов
Predicate<String> pred1 =  "1"::equals;
Predicate<String> pred2 =  "2"::equals;

public void tester(){
      Arrays.asList("1","2","3").stream().filter(pred1.or(pred2)).count();
}

Вы можете переместить ваши отдельные условия, чтобы сделать возможным их рекомбинацию другими способами.

0
Dale krebbers