java - 使用泛型和继承来谓词 java 8

标签 java generics java-8 predicate

我有三个 Foo Boo Goo 对象,其中包含 Foo 父类(super class)的 Boo 和 Goo 子类型。

我正在使用谓词来过滤列表,如下所示

public class Predicates {

        public static Predicate<Foo> isBoo() {
            foo ->  { 
                  Boo boo = (Boo) foo; 
                  return boo.isBoo();
             };
         }

    public static Predicate<Goo> isGoo() {
            foo ->  { 
                  Goo goo = (Goo) foo; 
                  return goo.isGoo();
             };
     }
}

接下来我想要实现的是使用泛型类型而不是指定的编译时类型。

public class Predicates {

        public static <T extends Foo> Predicate<T> isBoo() {
            foo ->  { 
                  // check both with instanceof
             };
         }

    public static <T extends Foo> Predicate<T> isGoo() {
            foo ->  { 
                  Goo goo = (Goo) foo; 
                  return boo.isGoo();
             };
     }

    public static <T extends Foo> Predicate<T> isFoo() {
            isGoo().or(isBoo());
     }
}

当另一个使用 OR 或 AND 组合多个谓词时,似乎不可能将泛型与谓词一起使用。知道谓词可能有不同的类型,但仍然扩展父类(super class)。

需要光!

最佳答案

为什么不为每个子类型定义多个谓词,将这些谓词添加到列表中,最后通过对列表中的所有谓词执行 or 操作来构造一个谓词。

public static Predicate<Foo> anotherIsFoo() {
    return predicates.stream()
            .reduce(foo -> false, (a, b) -> a.or(b));
}

上面的代码假设 predicates 是包含每个子类型的所有谓词的列表。

完整的代码如下所示:

import java.util.List;
import java.util.function.Predicate;
import java.util.stream.Collectors;

class Foo {}
class Boo extends Foo {}
class Goo extends Foo {}

public class Predicates {
    public static List<Predicate<Foo>> predicates = List.of(isGoo(), isFoo());

    public static Predicate<Foo> isGoo() {
        return foo -> foo instanceof Goo;
    }

    public static Predicate<Foo> isBoo() {
        return goo -> goo instanceof Boo;
    }

    public static Predicate<Foo> anotherIsFoo() {
        return predicates.stream()
            .reduce(foo -> false, (a, b) -> a.or(b));
    }

    public static Predicate<Foo> isFoo() {
        return foo -> foo instanceof Foo;
    }

    public static void main(String[] args) {
        List<Foo> list = List.of(new Boo(), new Goo(), new Goo(), new Goo());
        List<Boo> boos = list.stream()
                .filter(Predicates.isBoo())
                .map(boo -> (Boo) boo)
                .collect(Collectors.toList());
        System.out.println(boos.size()); // 1

        List<Goo> goos = list.stream()
                .filter(Predicates.isGoo())
                .map(goo -> (Goo) goo)
                .collect(Collectors.toList());
        System.out.println(goos.size()); //3

        List<Foo> foos = list.stream()
                .filter(Predicates.isFoo())
                .collect(Collectors.toList());
        System.out.println(foos.size()); // 4

        List<Foo> foos1 = list.stream()
                .filter(Predicates.anotherIsFoo())
                .collect(Collectors.toList());
        System.out.println(foos1.size()); // 4
    }
}

上面的代码回答了您的查询 - “即使它们继承自同一个父类(super class),我也不能在同一个列表中拥有两种不同的类型!”

以上代码需要 Java 9 或更高版本。

关于java - 使用泛型和继承来谓词 java 8,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55497012/

相关文章:

JavaFx,FXML,如何根据其 parent 自动调整内容大小: Button, StackPane,Field,Scrollpane

java - 让 edittext 显示与按下的键不同的键

c# - 泛型约束内的泛型类型参数

typescript - Typescript 中的通用方法

java - 需要帮助理解此通用声明

java-8 - 使用 Transformer 的值丰富 header

java - 如何修复使用 lambda 和条件从列表中删除重复项

spring - Maven 项目无法构建

java - 保存变量以在类中使用

java - 在 Netbeans 6.9 中进行 J2ME 开发需要哪些插件?