本系列参考书籍 : 《揭秘Java 虚拟机-JVM设计原理与实现》
推荐预备技能: 掌握C++函数指针,汇编call调用即可~
0 - 从Java字节码走向汇编
先看一段代码
#include<stdio.h>
int add(int a, int b){
return a+b;
}
int main(int argc, char* argv[]){
int a = 5;
int b = 3;
int ret = add(a,b);
printf("r = %d\n", ret);
return 0;
}
#include<stdio.h>
const unsigned char code[] = "\x55\x89\xe5\x8b\x45\x0c\x8b\x55\x08\x01\xd0\x5d\xc3";
int main(int argc, char* argv[]){
int a = 5;
int b = 3;
int (*fun)(int, int);
fun = (int (*)(int, int)) code;
int ret = fun(a,b);
printf("r = %d\n", ret);
return 0;
}
code[] 对应如下
两个代码运行结果一样
mac catalina版本如下:
#include<stdio.h>
const unsigned char code[] = "\x55\x48\x89\xe5\x89\x7d\xfc\x89\x75\xf8\x8b\x45\xfc\x03\x45\xf8\x5d\xc3";
int main(){
int a= 3;
int b = 2;
int ret = ((int(*)(int,int))code)(a,b);
printf("ret %d", ret);
return 0;
}
从结果来看,我们已经成功的能在C语言中直接 动态 执行机器码了.我们只要将Java字节码翻译 位汇编对应的机器码(总所周知这两个还挺像的)然后通过 类似函数调用的方式,不就能直接调用Java方法了吗?
是的,现代的JVM的确就是这么干的,但是又不完全,请看图
JVM 几乎采用了动态生成机器码,直接人工编写汇编等方法来优化Java的运行!
一般函数调用需要出栈入栈等操作(约7行汇编),而人工直接编写的却不需要(约3行汇编)甚至在android等特定系统字节码在javac期间便能转化为汇编!
本章完结,JVM最核心部分--run engine(执行引擎)已经大概了解了.
下一章Java执行引擎工作原理:方法调用.
本系列参考《揭秘Java 虚拟机-JVM设计原理与实现》,侵权删