我猜这是一个基本的问题,但是为什么没有sload指令呢?为什么你可以加载除了short之外的所有原语?(有saload,但仍然……)
对于:
public class ShortTest {
public void test() {
short i = 1;
System.out.print(i);
}
}
请参考JVM规范§2.11.1。类型和Java虚拟机:
请注意,表2.11.1-A中的大多数指令没有整数类型byte
、char
和short
的形式。都没有Boolean
类型的表单。编译器使用Java虚拟机指令对byte
和short
类型的文字值加载进行编码,这些指令在编译时或运行时将这些值签名扩展为int
类型的值。boolean
和char
类型的文字值的加载使用指令进行编码,这些指令在编译时或运行时将文字扩展为int
类型的值。同样,boolean
、byte
、short
和char
类型的值数组的加载使用Java虚拟机指令进行编码,这些指令将值签名扩展或零扩展为int
类型的值。因此,对实际类型boolean
、byte
、char
和short
值的大多数操作都是由对计算类型int
值操作的指令正确执行的。
值得回顾的是,在Java中,任何不涉及long
的整数算术都有int
结果,而不管输入是byte
、char
、short
还是int
。
所以一条线就像
short i = 1, j = 2, k = i + j;
不会编译,但需要类型强制转换,如
short i = 1, j = 2, k = (short)(i + j);
0: iconst_1
1: istore_1
2: iconst_2
3: istore_2
4: iload_1
5: iload_2
6: iadd
7: i2s
8: istore_3
的编译形式相同
int i = 1, j = 2, k = (short)(i + j);
但是请注意,在重载的情况下,变量的编译时类型可以改变编译器为调用选择的方法。如果类型带有不同的语义,这一点尤其重要,比如print(boolean)
或print(car)
。虽然在这两种情况下传递给方法的值都具有int
类型,但结果完全不同。
编译器实施的差异的另一个示例是
{
int i = 1;
i++;
}
{
short s = 1;
s++;
}
0: iconst_1
1: istore_1
2: iinc 1, 1
5: iconst_1
6: istore_1
7: iload_1
8: iconst_1
9: iadd
10: i2s
11: istore_1
因此,由于计算总是在32位中执行,编译器插入必要的代码,将结果截断为short
以进行第二次增量。再次注意没有变量声明,因此代码与
int i = 1;
i++;
i = 1;
i = (short)(i+1);
验证类型系统也值得一看,因为验证器将检查从局部变量和到局部变量的所有传输的有效性:
类型检查器基于验证类型的层次结构强制类型系统,如下所示。
Verification type hierarchy:
top
____________/\____________
/ \
/ \
oneWord twoWord
/ | \ / \
/ | \ / \
int float reference long double
/ \
/ \_____________
/ \
/ \
uninitialized +------------------+
/ \ | Java reference |
/ \ | type hierarchy |
uninitializedThis uninitialized(Offset) +------------------+
|
|
null
因此,与Java语言类型相比,类型系统被简化了,验证者并不介意,例如,如果将boolean
值传递给需要char
的方法,因为两者都是int
类型。
所以我和一位面试官(你的标准人力资源经理)谈了他们在面试中真正想要的东西,他提到了一些关于测试JVM指令知识的东西(以及其他很多东西)。 我不知道那是什么,但我以为他说的是字节码指令。 我做了一些研究,似乎在“JVM指令”上找不到任何东西,我只找到了字节码指令的引用,我已经知道了字节码指令。 例如:http://docs.oracle.com/javase/specs/jvms/se7/html/
本文向大家介绍JVM常用指令速查表,包括了JVM常用指令速查表的使用技巧和注意事项,需要的朋友参考一下 JVM 基本指令 基本指令集是最常用的,总结如下: 指令 释义 iconst_1 int型常量值1进栈 bipush 将一个byte型常量值推送至栈顶 iload_1 第二个int型局部变量进栈,从0开始计数 istore_1 将栈顶int型数值存入第二个局部变量,从0开始计数 iadd 栈顶两
问题内容: 是否有Java语言或JVM中的任何软件预取指令,例如__builtin_prefetch(在GCC中可用) 问题答案: 一件有趣的事是Hotspot JVM实际上 确实支持 预取! 它把和方法作为内部函数,并将其编译成相应的CPU指令。 不幸的是,没有声明这种方法。但是,如果将以下方法添加到Unsafe.java中,对其进行重新编译并替换rt.jar中的Unsafe.class(或仅添
我参考了下面的oracle jvm文档 https://docs.oracle.com/javase/specs/jvms/se7/html/jvms-6.html#jvms-6.5.pop 弹出操作弹出操作数堆栈顶部的值 总体安排 pop表格pop=87(0x57) 操作堆栈...,值→ ... 说明从操作数堆栈中弹出顶部值。 除非值是1类计算类型的值(§2.11.1),否则不得使用pop指令。
问题内容: 我正在阅读此博客文章。 作者正在谈论在多线程环境中打破in 。 有了: 变成: 作者说,我引用: “我在这里所做的是添加一个附加读取: 哈希的第二次读取,在返回之前 。听起来很奇怪,而且不太可能发生,第一次读取可以返回正确计算出的哈希值,内存模型允许第二次读取返回0!这是允许的,因为该模型允许对操作进行广泛的重新排序。第二次读取实际上可以在代码中移动,以便处理器在第一次读取之前进行处理