前言
说来话长,最近又闲的无聊,在看JDK的源码,但是很多关键的地方都是native方法,这就导致需要在往深处看,也就是需要看openjdk源码了,但是c++代码又谈何容易,况且也不怎么会。
但是想来想去,决定还是要研究一下的,在以前的文章中已经编译过了openjdk11,虽然过程坎坷,但也是成功了,那么接下来就是导入openjdk源码到ide中,在下面会以clion作为示例,clion下载安装就不说了。
激动的心,颤抖的手,Deepin下成功编译OpenJdk11!!!
光这导入就花了我1天,毕竟openjdk很复杂,而且参考的文章有的说只导入hotspot目录,有的说导入openjdk\src目录,还有的说要导入根目录,导来导去,差点没放弃,但是放弃这1天就浪费了,所以瞎搞了一顿也不知道算不算成功,反正能正常调试了。
后面会使用到编译好的java程序(指的是bin下面的java),所以必须先编译openjdk。
导入openjdk
然后在"选择"对话框中选择openjdk的根目录,没错就是根目录,然后见到ok点ok,见到finish点finish。
导入后大概这个样子。
然后按照下面gif进行配置,注意Executable是选择编译好的java可执行程序,在bin目录下,也就你编译出来的,不能选在官网下载的。
记住如果在Before launch下有东西的话全部移除。
点击运行就能出来版本信息,这就是上步配置的-version参数起的作用。
运行java程序
但是需要运行我们写好的程序才行啊,不然图了个啥,所以先简单写一个:
public class Main {
public static void main(String[] args){
int a=12;
int b=34;
System.out.println("a+b="+(a+b));
}
public static void test(String[] args){
System.out.println("test");
}
}
我们知道java的程序是从main方法开始的,但是能不能不从main开始,当然可以,就需要我们修改openjdk源码,上面的test方法稍后会充当main方法运行。
先进行编译。
然后重新配置一下启动参数。
再次运行。
上面说了,我们能不能不从main开始,当然可以,回到openjdk源码,找到java.c文件,定位到JavaMain方法下,里面有一段是获取main方法的方法id,接着后面会调用这个方法,我们只需要修改GetStaticMethodID参数中的"main"即可,然后重新编译,这次编译只需要执行make就行,而且速度很快。
关于([Ljava/lang/String;)V
的意思在以前分析class文件格式的时候也说过了,代表是一个String数组,并且方法是无返回值的。
int JNICALL
JavaMain(void * _args)
{
mainID = (*env)->GetStaticMethodID(env, mainClass, "main",
"([Ljava/lang/String;)V");
(*env)->CallStaticVoidMethod(env, mainClass, mainID, mainArgs);
}
再次运行,可以看到,这次没执行main方法,而是执行了test方法。
剩下就是打断点进行调试,比如线程的start0方法,最终会调用操作系统的api启动线程,我们可以从中进行断点调试。