出于调试目的,我试图在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]}
的东西,我认为这并不一定比您手动给它们一个描述更清晰,但对于不受您控制的谓词可能有用。