1. 兼容性实现
java语言是通过中间语言实现兼容的,java程序被编译后,生成中间语言,中间语言指令由java虚拟机负责解释和执行,虚拟机在运行时将中间语言实时翻译成对应底层的机器指令并运行;.class即为java的中间语言
2. 那么如何将中间语言翻译成机器指令呢?
我们先来了解下cpu执行代码的原理。要让cpu执行一段代码,只需要将CS:IP 段寄存器执行对应的代码片入口即可;这里我们需要简单了解下CS与IP ,他们是CPU内部的两个寄存器,CS保存段地址,IP保存偏移地址,两者可以唯一确定内存中的一个地址。CPU在执行执行之前会通过这两个寄存器定位目标内存位置,然后将位置中的机器指令取出来进行运算。函数跳转的本质其实就是修改这两个寄存器的内容,使其执行目标函数的内存首地址。java虚拟器要想让CPU执行对应的目标机器指令,那么也得修改这两个寄存器内容。
说到底,cpu只认识机器指令,编译器编译之后交给虚拟机的java的中间语言,cpu无法执行中间语言,虚拟机提供一种机制,将中间语言(字节码) 转化为本地机器指令。这个中间语言其实就是JVM定义的java字节码指令集。java的指令集总数不超过255个。而编译器的工作就是把java语言编译成字节码指令集,JVM 的作用就是实时将字节指令集转化为对应的机器指令并执行
3. 字节码指令集
JVM内存可分为操作数栈、局部变量表、堆、常量池、方法区,java执行逻辑的主战场是操作数栈,不管数据存放在堆栈,或者常量池,最终都会传送到操作数栈中,执行运算;CPU执行逻辑的主战场是寄存器。可以拿打仗来通俗的描述JVM执行逻辑很容易理解。假设主战场在A战区,为了作战,指挥官会不断的把军队、枪支弹药、后勤物资等运送到A区,指挥官发送一道道命令调动军队等开始干架,干完之后又发送命令调离军队等。
java的字节码指令集包含以下分类(不绝对)
- 数据交换指令:比如iload、istore等
- 函数调用指令:invokevirtual、invokestatic、return等,函数调用时压入一个栈帧,返回时弹出一个栈帧
- 运算指令集:iadd、isub、fadd、ddiv等
- 控制转移指令:流程判断
- 对象创建和类型转换指令:new等
如何查看java文件的字节码
javap.exe -c Main.java
Idea 配置过程 (使用 右键->external tools -> Show Byte Code)
其实java的字节码指令集也大概和汇编语言指令集有着类似的分类关系,下面也简单介绍下汇编指令
4. 汇编指令
- 数据传送指令:mov、pop等
- 算术运算指令:add、inc等
- 逻辑运算指令:与(add)、或、非、左移(shl)、右移等
- 串指令:字符操作
- 程序转移指令:jmp跳转、loop循环等
5. 总结
明白java程序如果最终被cpu执行,中间语言及JVM的执行策略。