当前位置: 首页 > 知识库问答 >
问题:

JVM指令-sload

万开畅
2023-03-14

我猜这是一个基本的问题,但是为什么没有sload指令呢?为什么你可以加载除了short之外的所有原语?(有saload,但仍然……)

对于:

public class ShortTest {
    public void test() {
        short i = 1;
        System.out.print(i);
    }
}

共有1个答案

吴俊晤
2023-03-14

请参考JVM规范§2.11.1。类型和Java虚拟机:

请注意,表2.11.1-A中的大多数指令没有整数类型bytecharshort的形式。都没有Boolean类型的表单。编译器使用Java虚拟机指令对byteshort类型的文字值加载进行编码,这些指令在编译时或运行时将这些值签名扩展为int类型的值。booleanchar类型的文字值的加载使用指令进行编码,这些指令在编译时或运行时将文字扩展为int类型的值。同样,booleanbyteshortchar类型的值数组的加载使用Java虚拟机指令进行编码,这些指令将值签名扩展或零扩展为int类型的值。因此,对实际类型booleanbytecharshort值的大多数操作都是由对计算类型int值操作的指令正确执行的。

值得回顾的是,在Java中,任何不涉及long的整数算术都有int结果,而不管输入是bytecharshort还是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!这是允许的,因为该模型允许对操作进行广泛的重新排序。第二次读取实际上可以在代码中移动,以便处理器在第一次读取之前进行处理