String s = new String("xyz") 创建了几个对象?
实例分析1
javac编译代码,然后用javap来反编译,执行javap -c Test
从结果来看,ldc命令在常量池中创建了一个"xyz"的对象,然后把他推至操作数栈顶,然后astore保存到局部变量,return返回。
实例分析2
同上反编译分析后
很明显,我们看到new 创建了一个String对象,同时ldc在常量池中创建了"xyz"字符串对象,之后invokespecial执行构造函数,astore_1赋值,return返回。
通过两个实例分析,可以知道String s = new String("xyz"); 创建了2个对象,而有些答案说的3个对象,则是把引用s也算作一个对象。还有答案说xyz存在就创建了2个,不存在就创建了3个(包含引用s),再来测试一下。
测试分析
同上反编译后
从这里,很明显的发现这就是我们例子1和2的一个结合,但是注意两次ldc后面的#2,#号代表着索引,说明第二次new String("xyz")的时候并没有重新创建xyz对象。
总结
到底创建了几个对象呢?
- 如果xyz不存在,引用算对象的话,那就是3个
- 如果xyz不存在,引用不算对象的话,那就是2个
- 如果xyz存在,引用算对象的话,那就是2个
- 如果xyz存在,引用不算对象的话,那就是1个
当然,我认为引用肯定是不算对象的,最终答案应该是1或者2个。
常见的指令助记符含义
-
nop, 什么都不做。
-
aconst_null,将 null 推送至栈顶。
-
iconst_i(变量数字),将 int 型 i 推送至栈顶。同理有lconst_0,fconst_0这种你应该知道什么意思了
-
ldc,将 int,float 或 String 型常量值从常量池中推送至栈顶。
-
iload,将指定的 int 型局部变量推送至栈顶。
-
istore,将栈顶 int 型数值存入指定局部变量。同理astore_i代表将栈顶引用型数值存入第i个局部变量。
-
dup,复制栈顶数值并将复制值压入栈顶。
-
invokevirtual,调用实例方法。
-
invokespecial,调用超类构造方法,实例初始化方法,私有方法。
-
invokestatic,调用静态方法。
-
invokeinterface,调用接口方法。
-
invokedynamic,调用动态链接方法。
-
new,创建一个对象,并将其引用值压入栈顶。
补充题目
运行结果为: