AFAIK, w niejawnych konstruktorach języka Java generowane są zawsze dla klasy bez konstruktorów [1], [2].Czy prawidłowe jest posiadanie klasy kodu bajtowego JVM bez żadnego konstruktora?
Ale w bajtodzie nie mogłem znaleźć takiego ograniczenia na JVMS.
Więc:
jest ważne zgodnie z JVMs zdefiniowanie klasy bez konstruktora tylko wykorzystać swoje metody statyczne, jak w poniższym Jasmin cześć świata?
ma to dalsze konsekwencje, oprócz tego, że nie można go utworzyć? Nie będę mógł używać
invokespecial
do inicjowania wystąpień, co czyni bezużytecznymnew
zgodnie z https://docs.oracle.com/javase/specs/jvms/se7/html/jvms-4.html#jvms-4.10.2.4 (nie można użyć niezainicjowanego obiektu).
kod Jasmin:
.class public Main
.super java/lang/Object
.method public static main([Ljava/lang/String;)V
.limit stack 2
getstatic java/lang/System/out Ljava/io/PrintStream;
ldc "Hello World!"
invokevirtual java/io/PrintStream/println(Ljava/lang/String;)V
return
.end method
czyli bez konstruktora:
.method public <init>()V
aload_0
invokenonvirtual java/lang/Object/<init>()V
return
.end method
?
Praca z java Main
daje oczekiwane wyjście Hello World!
.
Sprawdziłem dane wyjściowe javap -v
iw przeciwieństwie do Java, jasmin
nie wygenerowałem domyślnego konstruktora.
Próbowałem również zadzwonić new Main();
w każdym razie, aby zobaczyć, co dzieje się z:
public class TestMain {
public static void main(String[] args) {
Main m = new Main();
}
}
i zgodnie z oczekiwaniami daje błąd kompilacji cannot find symbol
. Jeśli dodaję konstruktora do jasminu, działa TestMain
.
Wyjście javap -v
dla kompletności:
public class Main
minor version: 0
major version: 46
flags: ACC_PUBLIC, ACC_SUPER
Constant pool:
#1 = Utf8 Main.j
#2 = Class #17 // Main
#3 = NameAndType #21:#23 // out:Ljava/io/PrintStream;
#4 = Utf8 ([Ljava/lang/String;)V
#5 = Utf8 java/lang/Object
#6 = Class #5 // java/lang/Object
#7 = Utf8 Hello World!
#8 = Class #16 // java/io/PrintStream
#9 = String #7 // Hello World!
#10 = Class #19 // java/lang/System
#11 = Utf8 Code
#12 = Utf8 main
#13 = Fieldref #10.#3 // java/lang/System.out:Ljava/io/PrintStream;
#14 = Utf8 SourceFile
#15 = NameAndType #18:#22 // println:(Ljava/lang/String;)V
#16 = Utf8 java/io/PrintStream
#17 = Utf8 Main
#18 = Utf8 println
#19 = Utf8 java/lang/System
#20 = Methodref #8.#15 // java/io/PrintStream.println:(Ljava/lang/String;)V
#21 = Utf8 out
#22 = Utf8 (Ljava/lang/String;)V
#23 = Utf8 Ljava/io/PrintStream;
{
public static void main(java.lang.String[]);
descriptor: ([Ljava/lang/String;)V
flags: ACC_PUBLIC, ACC_STATIC
Code:
stack=2, locals=1, args_size=1
0: getstatic #13 // Field java/lang/System.out:Ljava/io/PrintStream;
3: ldc #9 // String Hello World!
5: invokevirtual #20 // Method java/io/PrintStream.println:(Ljava/lang/String;)V
8: return
}
SourceFile: "Main.j"
Jeśli ktoś może generować że z javac (w szczególności nie ACC_INTERFACE
ani ACC_SYNTHETIC
), że byłby to dobry argument za ważności.
Czy próbowałeś napisać obiekt Java, który wywołuje 'new Main();'? – RealSkeptic
@RealSkeptic po prostu zrobił i dostał 'error: can not find symbol' zgodnie z oczekiwaniami. Następnie, jeśli dodaję konstruktor do Jasmin, działa 'new Main()'. –
Cóż, więc. Twoje pytanie jest mniej lub bardziej filozoficzne. Czy to jest poprawna Java? Nie. Czy to prawda Jasmin? Tak. Czy można go użyć z programu Java? Tak. Czy działa tak, jak ludzie oczekują, że klasa Java będzie działać? Nie do końca. Czym jest "Ważny"? – RealSkeptic