JDK ist Oracle 'JDK 1.8u65, aber das Problem wurde mit "so niedrig wie" 1.8u25 auch gesehen. HierEnum, Schnittstellen und (Java 8) Lambdas: Code kompiliert, schlägt aber zur Laufzeit fehl; wird das erwartet?
ist die volle SSCCE:
public final class Foo
{
private interface X
{
default void x()
{
}
}
private enum E1
implements X
{
INSTANCE,
;
}
private enum E2
implements X
{
INSTANCE,
;
}
public static void main(final String... args)
{
Stream.of(E1.INSTANCE, E2.INSTANCE).forEach(X::x);
}
}
Dieser Code kompiliert; aber es schlägt zur Laufzeit:
Exception in thread "main" java.lang.BootstrapMethodError: call site initialization exception
at java.lang.invoke.CallSite.makeSite(CallSite.java:341)
at java.lang.invoke.MethodHandleNatives.linkCallSiteImpl(MethodHandleNatives.java:307)
at java.lang.invoke.MethodHandleNatives.linkCallSite(MethodHandleNatives.java:297)
at com.github.fge.grappa.debugger.main.Foo.main(Foo.java:38)
at sun.reflect.NativeMethodAccessorImpl.invoke0(Native Method)
at sun.reflect.NativeMethodAccessorImpl.invoke(NativeMethodAccessorImpl.java:62)
at sun.reflect.DelegatingMethodAccessorImpl.invoke(DelegatingMethodAccessorImpl.java:43)
at java.lang.reflect.Method.invoke(Method.java:497)
at com.intellij.rt.execution.application.AppMain.main(AppMain.java:144)
Caused by: java.lang.invoke.LambdaConversionException: Invalid receiver type class java.lang.Enum; not a subtype of implementation type interface com.github.fge.grappa.debugger.main.Foo$X
at java.lang.invoke.AbstractValidatingLambdaMetafactory.validateMetafactoryArgs(AbstractValidatingLambdaMetafactory.java:233)
at java.lang.invoke.LambdaMetafactory.metafactory(LambdaMetafactory.java:303)
at java.lang.invoke.CallSite.makeSite(CallSite.java:302)
... 8 more
Es in Code zu reparieren ist "einfach"; im Hauptverfahren, müssen Sie nur noch auf:
// Note the <X>
Stream.<X>of(E1.INSTANCE, E2.INSTANCE).forEach(X::x);
EDIT in der Tat gibt es eine zweite Art und Weise, wie sie in der akzeptierten Antwort erwähnt ... Ersetzen Sie die Methode Referenz mit einem Lambda:
Stream.of(E1.INSTANCE, E2.INSTANCE).forEach(x -> x.x());
Also, uh. was geschieht hier? Warum kompiliert der ursprüngliche Code überhaupt? Ich hätte erwartet, dass der Compiler bemerkt, dass die Methodenreferenz nicht auf irgendwas war Enum<?>
, aber auf X
, aber nein ...
Was vermisse ich? Ist das ein Fehler im Compiler? Ein Missverständnis von mir?
Vielleicht ein verwandter 'javac'-Fehler: [JDK-8141508] (https://bugs.openjdk.java.net/browse/JDK-8141508), obwohl er sich mit sich überschneidenden Typen zu befassen scheint. – Tunaki
@Tunaki interessant; Ich werde dieses Problem durchlesen, vielleicht ist es das gleiche ... – fge
Es * ist * ein sich überschneidender Typ. Der abgeleitete Typ von "Stream.of (E1.INSTANCE, E2.INSTANCE)" ist laut meiner Eclipse "Stream & Foo.X>". –
RealSkeptic