创建lambda表达式的字符串表示形式



出于调试目的,我试图在Java 8中创建lambda表达式的字符串表示(特别是Predicate 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()没有被覆盖)。

我不确定这样的事情是否可能,例如,通过反思,我到目前为止当然还没有找到任何关于它的东西。什么好主意吗?

我能想到的最简单的事情是创建一个"命名谓词",为您的谓词提供名称或描述,基本上是任何将作为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]}的东西,我认为这并不一定比您手动给它们一个描述更清晰,但对于不受您控制的谓词可能有用。

相关内容

  • 没有找到相关文章