前提

对于springboot的类加载原理和spring-boot-loader的源码解析,网上已有很多的很棒的文章了。我始终置信对于技术原理的获取,代码层面的话肯定是动眼比不上入手,debug跟着源码运行走一遍,成果往往是很棒的。同理,对于spring-boot-loader的原理,源码的把握,其原理和debug形式参考:springboot jar包可运行,debug通知你怎么运行的

本文想要讲述的可能要更近一步了。带着一些疑难开始咱们的学习

  • 1、程序是从哪进入JarLauncher.main()办法的
  • 2、java -jar xxx-executable.jar java有什么规定吗
  • 3、jvm层面是如何进入到java程序的,连接点在哪里
》本文力求专一和精简,心愿你有所播种和想法

注释

-jar的规矩

其实,java-jar定义了一些规定,只有合乎这个规定,你做啥都行,怎么做都行。其中的一个是:jar中须要一个META-INF/MAINFEST.MF文件,且文件中Main-Class manifest header。官网的阐明如下

所以,咱们解压一个xxx-executable.jar,会看到META-INF/MAINFEST.MF的文件及其内容

正是因为有了Main-Class: org.springframework.boot.loader.JarLauncher,所以,当java -jar xxx-executable.jar时,程序会进入org.springframework.boot.loader.JarLauncher执行main办法。到这里我又有了疑难,那是从哪进入JarLauncher.main办法的呢

程序是怎么进入JarLauncher.main办法的呢

这个时候想到了jvm。应该是从jvm进去到了JarLauncher.main,那么怎么验证呢。

咱们通过CLion查看jvm源码,试图找到这个答案。咱们通过全局搜寻Main-Class,Manifest 这些关键词来定位所在的文件,并打上相应的debug端点。如下图

而后咱们启动debug模式启动jvm,启动时配置上咱们的利用jar包:-jar xxx-executable.jar,如下图

当初咱们跟着jvmdebug一起追下寻找Main-Class,Manifest 的过程。

看到了JLI_ParseManifest,从名字也看的进去这个头绪了

能够看到static const char *manifest_name = "META-INF/MANIFEST.MF"在jvm中是个常量。所以,咱们的利用xxx-executable.jar中必须有个叫这个名的目录和文件。办法接着解析了这个文件,文件的内容和咱们解压xxx-executable.jar看到的内容统一,如下图

所以,到这Main-Class的值就被解析进去了。上面咱们看下jvm如何加载Main-Class的值(即org.springframework.boot.loader.JarLauncher)的。

上面图中6个局部其实将虚拟机的整个初始化和加载过程都显示进去了。在这里咱们只关注JarLauncher是如何加载初始化与JarLauncher.main是怎么运行的,所以咱们重点关注(1)、(2)、(4)、(6)

(1)LoadMainClass - 加载main class

咱们先看下LoadMainClass的细节,看看她咋加载的。通过GetLauncherHelperClass办法,咱们终于看到了相熟的身影:sun/launcher/LauncherHelper 这个java class。如下图

LoadMainClass做了三件事:
1、获取LauncherHelper实例
2、通过checkAndLoadMain办法应用ClassLoader.getSystemClassLoader初始化org.springframework.boot.loader.JarLauncher class
3、makePlatformString获取utf-8后的string

须要特地指出的是LauncherHelper.java。这个类的源码在jvm中,在咱们相熟的rt.jar是以.class的模式呈现的。我把他俩放在一张图片里比照下

(2)获取LauncherHelper的实例

这里的实例是在(1)中初始化好的,用就好了

(4)验证和加载main办法

这里用了相似反射的办法来获取main办法

(6)调用LauncherHelper.main办法

这里就是咱们始终要晓得的那个中央,也是文章结尾局部的问题1和3的答案。即jvm运行和java application的连接点。利用CLionIdea,利用debug,我人为的把他俩的运行联合在一起,具象的示意出这两个点的运行,如下图

it`s time to summary

到这里,咱们的疑难都找到答案了。咱们从jvm源码到java代码,整个流程串下来。置信有人再问你springboot的入口时,你不仅晓得了是JarLauncher.main,而且能进一步晓得从哪(怎么)进入的JarLauncher.main的。这不论是对工作,还是面试,对你都是有很大收益的。

原味地址:jvm怎么运行springboot jar文件的

附录

所有,从jvmspringboot,再到利用app的入口,咱们当初都晓得了。

java.c的JavaMain办法 --> LauncherHelper.java的main办法 --> Application.java的main办法