11

出于调试目的,我正在尝试Predicate在 Java 8 中创建 lambda 表达式(特别是 s,尽管它对其他 lambda 表达式也很有趣)的字符串表示形式。我的想法是这样的:

public class Whatever {

    private static <T> String predicateToString(Predicate<T> predicate) {
        String representation = ... // do magic
        return representation;
    }

    public static void main(String[] args) {
        System.out.println(Whatever.<Integer>predicateToString(i -> i % 2 == 0));
    }

}

并且输出将是i -> i % 2 == 0(或逻辑上等效的)。该toString()方法似乎没有帮助,输出就像com.something.Whatever$$Lambda$1/1919892312@5e91993f(我想这是可以预期的,因为toString()它没有被覆盖)。

我不确定这样的事情是否可能,例如通过反射,到目前为止我当然无法找到任何东西。有任何想法吗?

4

1 回答 1

21

我能想到的最简单的事情是创建一个“命名谓词”,为您的谓词提供名称或描述,基本上任何有用的东西toString

public class NamedPredicate<T> implements Predicate<T> {
    private final String name;
    private final Predicate<T> predicate;

    public NamedPredicate(String name, Predicate<T> predicate) {
        this.name = name;
        this.predicate = predicate;
    }

    @Override
    public boolean test(T t) {
        return predicate.test(t);
    }

    @Override
    public String toString() {
        return name;
    }

    public static void main(String... args) {
        Predicate<Integer> isEven = new NamedPredicate<>("isEven", i -> i % 2 == 0);
        System.out.println(isEven); // prints isEven
    }
}

可以说,像这样给你的谓词命名或描述会使你使用它们的代码更容易理解:

Stream.of(1, 2, 3, 4)
        .filter(isEven)
        .forEach(System.out::println);

一个奇怪的想法可能是推导出谓词的“结构”描述,即某些给定输入的输出是什么?显然,当输入集是有限且小的(例如,对于枚举、布尔值或其他一些受限集)时,这将最有效,但我想你也可以为整数谓词尝试一小组“随机”整数:

private static Map<Boolean, List<Integer>> testPredicate(Predicate<Integer> predicate) {
    return Stream.of(-35, -3, 2, 5, 17, 29, 30, 460)
            .collect(Collectors.partitioningBy(predicate));
}

对于isEven,这将返回类似{false=[-35, -3, 5, 17, 29], true=[2, 30, 460]}的内容,我认为这不一定比您手动给它们描述更清楚,但对于不受您控制的谓词可能有用。

于 2014-05-16T21:56:39.723 回答