假设我有一个使用lambda表达式(闭包)定义的对象列表。有没有办法检查它们,以便比较它们?有没有办法比较lambdas?
我最感兴趣的代码是
List<Strategy> strategies = getStrategies();
Strategy a = (Strategy) this::a;
if (strategies.contains(a)) { // ...
完整的代码
import java.util.Arrays;
import java.util.List;
public class ClosureEqualsMain {
interface Strategy {
void invoke(/*args*/);
default boolean equals(Object o) { // doesn't compile
return Closures.equals(this, o);
}
}
public void a() { }
public void b() { }
public void c() { }
public List<Strategy> getStrategies() {
return Arrays.asList(this::a, this::b, this::c);
}
private void testStrategies() {
List<Strategy> strategies = getStrategies();
System.out.println(strategies);
Strategy a = (Strategy) this::a;
// prints false
System.out.println("strategies.contains(this::a) is " + strategies.contains(a));
}
public static void main(String... ignored) {
new ClosureEqualsMain().testStrategies();
}
enum Closures {;
public static <Closure> boolean equals(Closure c1, Closure c2) {
// This doesn't compare the contents
// like others immutables e.g. String
return c1.equals(c2);
}
public static <Closure> int hashCode(Closure c) {
return // a hashCode which can detect duplicates for a Set<Strategy>
}
public static <Closure> String asString(Closure c) {
return // something better than Object.toString();
}
}
public String toString() {
return "my-ClosureEqualsMain";
}
}
这样看来,唯一的解决办法是定义每个lambda作为一个字段,只使用这些字段。如果您想打印出所调用的方法,最好使用Method
。用lambda表达式有更好的方法吗?
另外,是否可以打印一个lambda并获得一些人类可读的东西?如果您打印this::a
,而不是
ClosureEqualsMain$$Lambda$1/[email protected]
得到类似
ClosureEqualsMain.a()
甚至使用this.toString
和方法。
my-ClosureEqualsMain.a();
您可以在闭包中定义toString,equals和hashhCode方法。 –
@AnkitZalani你可以给一个编译的例子吗? –
@PeterLawrey,由于在Object中定义了toString,所以我认为你可以定义一个接口来提供'toString'的默认实现,而不会违反* single-method *接口的功能。我没有检查过这个。 –