【发布时间】:2015-04-06 19:24:30
【问题描述】:
AFAIK,在 Java 中,总是为没有构造函数 [1]、[2] 的类生成隐式构造函数。
但在字节码中我找不到对JVMS 的这种限制。
所以:
根据JVMS定义一个没有构造函数的类只使用它的静态方法是否有效,如下面的jasmin hello world?
除了不能创建它的实例之外,它还有其他后果吗?我将无法使用
invokespecial来初始化实例,这使得new根据https://docs.oracle.com/javase/specs/jvms/se7/html/jvms-4.html#jvms-4.10.2.4 无用(不能使用未初始化的对象)。
茉莉码:
.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
也就是说,没有构造函数:
.method public <init>()V
aload_0
invokenonvirtual java/lang/Object/<init>()V
return
.end method
?
使用java Main 运行会得到预期的输出Hello World!。
我检查了javap -v 的输出,与Java 不同,jasmin 没有生成默认构造函数。
我也试过打电话给new Main(); 看看会发生什么:
public class TestMain {
public static void main(String[] args) {
Main m = new Main();
}
}
正如预期的那样,它给出了编译错误cannot find symbol。如果我将构造函数添加到 jasmin,那么 TestMain 可以工作。
为了完整性,输出javap -v:
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"
如果有人可以使用 javac 生成它(特别是没有 ACC_INTERFACE 或 ACC_SYNTHETIC),那将是一个很好的有效性论据。
【问题讨论】:
-
您是否尝试过编写调用
new Main();的Java 对象? -
@RealSkeptic 刚刚做了,并按预期得到了
error: cannot find symbol。然后,如果我将构造函数添加到 Jasmin,new Main()就可以了。 -
好吧,那么。那么你的问题或多或少是哲学的。它是有效的Java吗?不,它是有效的 Jasmin 吗?是的。可以从 Java 程序中使用吗?是的。它是否像人们期望的那样工作 Java 类?不是完全。那么,什么是“有效”?
-
@RealSkeptic 1) 根据 JVMS 有效。它在我的实现上运行的事实并不能保证 2) 除了无法创建实例之外,还会发生其他不好的事情吗?
标签: java constructor jvm bytecode