Ich versuche, LambdaMetafactory.metafactory explizit zu verwenden, ich kann nicht verstehen, warum es nur mit der funktionsfähigen Runnable-Schnittstelle funktioniert. Für das Beispiel tut dieser Code, was es zu erwarten ist (es druckt „Hallo Welt“):Explizite Verwendung von LambdaMetafactory
public class MetafactoryTest {
public static void main(String[] args) throws Throwable {
MethodHandles.Lookup caller = MethodHandles.lookup();
MethodType methodType = MethodType.methodType(void.class);
MethodType invokedType = MethodType.methodType(Runnable.class);
CallSite site = LambdaMetafactory.metafactory(caller,
"run",
invokedType,
methodType,
caller.findStatic(MetafactoryTest.class, "print", methodType),
methodType);
MethodHandle factory = site.getTarget();
Runnable r = (Runnable) factory.invoke();
r.run();
}
private static void print() {
System.out.println("hello world");
}
}
Das Problem entsteht, wenn ich versuche, eine andere funktionale Schnittstelle, wie zB Lieferanten zu verwenden. Der folgende Code funktioniert nicht:
public class MetafactoryTest {
public static void main(String[] args) throws Throwable {
MethodHandles.Lookup caller = MethodHandles.lookup();
MethodType methodType = MethodType.methodType(String.class);
MethodType invokedType = MethodType.methodType(Supplier.class);
CallSite site = LambdaMetafactory.metafactory(caller,
"get",
invokedType,
methodType,
caller.findStatic(MetafactoryTest.class, "print", methodType),
methodType);
MethodHandle factory = site.getTarget();
Supplier<String> r = (Supplier<String>) factory.invoke();
System.out.println(r.get());
}
private static String print() {
return "hello world";
}
}
Exception in thread "main" java.lang.AbstractMethodError: metafactorytest.MetafactoryTest$$Lambda$1/258952499.get()Ljava/lang/Object;
at metafactorytest.MetafactoryTest.main(MetafactoryTest.java:29)
Sollte nicht die beide Code-Snippet Arbeit in einer ähnlichen Art und Weise, die das Problem in dem zweiten Code-Snippet ist?
Darüber hinaus ist der folgende Code, dass gleichwertig sein sollte, funktioniert gut:
public class MetafactoryTest {
public static void main(String[] args) throws Throwable {
Supplier<String> r = (Supplier<String>)() -> print();
System.out.println(r.get());
}
private static String print() {
return "hello world";
}
}
bearbeitet
andere Lösung, die der Methode Rückgabetyp zu ändern vermeidet, ist eine neue funktionale Schnittstelle zu definieren:
public class MetafactoryTest {
@FunctionalInterface
public interface Test {
String getString();
}
public static void main(String[] args) throws Throwable {
MethodHandles.Lookup caller = MethodHandles.lookup();
MethodType methodType = MethodType.methodType(String.class);
MethodType invokedType = MethodType.methodType(Test.class);
CallSite site = LambdaMetafactory.metafactory(caller,
"getString",
invokedType,
methodType,
caller.findStatic(MetafactoryTest.class, "print", methodType),
methodType);
MethodHandle factory = site.getTarget();
Test r = (Test) factory.invoke();
System.out.println(r.getString());
}
private static String print() {
return "hello world";
}
Vielleicht ist das Problem mit dem Methodennamen "run" übergeben Sie als zweites Argument. Runnable hat eine "run" -Methode. Lieferant nicht. – Eran
Das war ein Fehler (Der Runnable Fall funktioniert nur mit "run"), aber auch mit get das zweite Snippet gibt diesen Fehler. – andrebask