初识JVM

时间:2024-01-11 10:32:08

做了这么久的开发,到目前为止对JVM也只是一些简单的概念上的理解,正好周末于是将JVM的学习提上日程。

JVM 概念

JVM是Java Virtual Machine(Java虚拟机)的缩写,JVM是一种用于计算设备的规范,它是一个虚构出来的计算机,是通过在实际的计算机上仿真模拟各种计算机功能来实现的。
        一般的高级语言如果要在不同的平台上运行,至少需要编译成不同的目标代码。而引入Java语言虚拟机后,Java语言在不同平台上运行时不需要重新编译。Java语言使用Java虚拟机屏蔽了与具体平台相关的信息,使得Java语言编译程序只需生成在Java虚拟机上运行的目标代码(字节码),就可以在多种平台上不加修改地运行。Java虚拟机在执行字节码时,把字节码解释成具体平台上的机器指令执行。这就是Java的能够“一次编译,到处运行”的原因。(摘录自百度百科)

JVM 规范

JVM 规范主要定义了二进制 class 文件 和 JVM 指令集等

- class 文件类型

- 运行时数据

- 帧栈

- 虚拟机的启动

- 虚拟机的指令集

  • 类型转化(l2i)
  • 出入栈操作(aload astore)
  • 运算(iadd isub)
  • 流程控制(ifeq ifne)
  • 函数调用(invokevirtual invokeinteeerface invokespespecial invokestatic)

- 整数的表达

  • 源码:第一位为符号位(0为正数,1为负数)
    7
    原码:00000111
  • 反码:符号位不动,原码区反
    6
    原码:00000110
    反码:01111001
  • 负数补码:符号位不动,反码加1
    -5
    原码:10000101
    反码:11111010
    补码:11111011
  • 正数补码:和原码相同
    5
    原码:00000101
    反码:01111010
    补码:00000101
  • 打印整数二进制
int a = 6;
for(int i = 0; i < 32; i++){
    int t = (a & 80000000 >>>i) >>> (31 - i);
    System.out.print(t);
}
第一次循环:
0000 0000 0000 0000 0000 0000 0000 0110 (6的原码)
1000 0000 0000 0000 0000 0000 0000 0000 (80000000 >>> 0 的值)
0000 0000 0000 0000 0000 0000 0000 0000 (6 & 80000000 >>> 0)
0000 0000 0000 0000 0000 0000 0000 0000 (上面结果向右位移31位的值)
第二次循环:
0000 0000 0000 0000 0000 0000 0000 0110 (6的原码)
0100 0000 0000 0000 0000 0000 0000 0000 (80000000 >>> 1 的值)
0000 0000 0000 0000 0000 0000 0000 0000 (6 & 80000000 >>> 1)
0000 0000 0000 0000 0000 0000 0000 0000 (上面结果向右位移30位的值)
                      .
                      .
                      .
第三十次循环:
0000 0000 0000 0000 0000 0000 0000 0110 (6的原码)
0000 0000 0000 0000 0000 0000 0000 0100 (80000000 >>> 29 的值)
0000 0000 0000 0000 0000 0000 0000 0100 (6 & 80000000 >>> 29)
0000 0000 0000 0000 0000 0000 0000 0001 (上面结果向右位移2位的值)
第三十一次循环:
0000 0000 0000 0000 0000 0000 0000 0110 (6的原码)
0000 0000 0000 0000 0000 0000 0000 0010 (80000000 >>> 30 的值)
0000 0000 0000 0000 0000 0000 0000 0010 (6 & 80000000 >>> 30)
0000 0000 0000 0000 0000 0000 0000 0001 (上面结果向右位移1位的值)
第三十二次循环:
0000 0000 0000 0000 0000 0000 0000 0110 (6的原码)
0000 0000 0000 0000 0000 0000 0000 0001 (80000000 >>> 31 的值)
0000 0000 0000 0000 0000 0000 0000 0000 (6 & 80000000 >>> 31)
0000 0000 0000 0000 0000 0000 0000 0000 (上面结果向右位移0位的值)

因为int值为32位,所以上述计算均为32位码计算,最后将每次循环所得的数打印连接就是6的二进制码

- 为什么要使用补码

  • 补码可以直接参与加法运算,符号位也可以直接参与运算,最后计算结果为正数则符号位是可以被进位进掉的
计算:-6+5=-1
11111010
00000101
11111111(-1的补码) 计算:-4+5=1
11111100
00000101
00000001
  • 表示0
    因为0既不是正数也不是负数,那么我们在计算机中怎么表示0呢?
0
正数表示:00000000
负数表示:10000000
正数补码:00000000
负数补码:00000000

结果发现0不论是表示为正数还是负数它的补码都是00000000

- 浮点数的表示

一个浮点数 (Value) 的表示其实可以这样表示:
也就是浮点数的实际值,等于符号位(sign bit)乘以指数偏移值(exponent bias)再乘以分数值(fraction)。
对于不同长度的浮点数,阶码与小数位分配的数量不一样,如下:

类型 数符 阶码 尾数 总位数 偏移值
短实数 1 8 23 32 127
长实数 1 11 52 64 1023
零时实数 1 15 64 80 16383
  • 我们对单精度浮点数为例,数符分配是1位,阶码分配了8位,尾数分配了是23位
  • 例如: -5.125
符号位:1
整数部分:00000101(除2求余法)
小数部分:001(乘2求整法)
整体则为:00000101.001
转化二进制浮点数,使得整数未为1   1.01001 * 10   10为二进制2,表示小数点位移2位
阶码:单精度的浮点数阶码为8,偏移值为01111111  则 10 + 01111111 = 10000001
尾数:01001
则二进制码为: 1 10000001 01001000000000000000000(数符 阶码 尾数)

- JVM 需要对java Library 提供以下支持

  • 反射 java.lang.reflect
  • ClassLoader
  • 初始化 class 和 interface
  • 安全相关 java.security
  • 多线程
  • 弱引用

- JVM的编译

  • 原码到JVM指令的对应格式
  • javap
  • JVM 反汇编格式
<index><opcode>[<operand1>[operand2]][<commend>]

for(i = 0; i < 100; i++){
}

上述代码对应的汇编语言为:

0 iconst_0     //Push int constant 0
1 istore_1     //Store into local variavle 1 (i=0)
2 goto 8       //Fist time through don`t increment
5 iinc 1 1     //Increment local variable 1 by 1(i++)
8 iload_1      //Push local variable 1 (i)
9 bipush 100   //Push int constant 100
11 if_icmplt 5 //Compare and loop if less than (i<100)
14 return      //Return void when done 今天也算是对jvm有一个初步的了解,后续将对jvm有一个更深层次的学习,如果有什么不对的地方希望各位大佬及时指正。

-------------------- END ---------------------

最后附上作者的微信公众号地址和博客地址

公众号:wuyouxin_gzh

初识JVM

Herrt灬凌夜:https://www.cnblogs.com/wuyx/