前言
说来话长,最近又闲的无聊,在看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程序
publicclassMain{ publicstaticvoidmain(String[] args){ inta=12; intb=34; System.out.println("a+b="+(a+b)); } publicstaticvoidtest(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数组,并且方法是无返回值的。
恕我直言,这可能是你见过最详细的class文件结构分析
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启动线程,我们可以从中进行断点调试。