欢迎您访问程序员文章站本站旨在为大家提供分享程序员计算机编程知识!
您现在的位置是: 首页  >  IT编程

三分钟带你了解SpringBoot真正的启动引导类

程序员文章站 2022-03-17 18:05:00
引言springboot项目中的启动类,一般都是xxapplication,例如**「statsapplication」,「unionapplication」**。每个项目的启动类名称都不一样。但是它...

引言

springboot项目中的启动类,一般都是xxapplication,例如**「statsapplication」「unionapplication」**。

每个项目的启动类名称都不一样。但是它的启动类真的是xxapplication吗?

三分钟带你了解SpringBoot真正的启动引导类

**meta-inf/**manifest.mf文件

jar文件实际上是class文件的zip压缩存档。jar并不能表达应用程序的便签信息.

「meta-inf/manifest.mf文件提供存档的便签信息.」

manifest.mf有 「main-class,用来标明jar文件的入口类。」

解压jar包,查看meta-inf/manifest.mf过程如下:

三分钟带你了解SpringBoot真正的启动引导类

重要信息如下

main-class: org.springframework.boot.loader.jarlauncher
start-class: com.shanyuan.statsapplication

也就是说:「org.springframework.boot.loader.jarlauncher 是 springboot 的启动类!」

下面浏览下 jarlauncher

浏览jarlauncher

3.1 找到jarlauncher

进入idea,ctrl+n查找jarlauncher,竟然找不到!!

三分钟带你了解SpringBoot真正的启动引导类

进入 https://search.maven.org/classic/#advancedsearch 查询jarlauncher

三分钟带你了解SpringBoot真正的启动引导类

在查询结果找到spring下的项目

三分钟带你了解SpringBoot真正的启动引导类

确定 jarlauncher 位于 spring-boot-loader 下。为了方便查看源码,在 pom 中引入

<dependency>
 <groupid>org.springframework.boot</groupid>
 <artifactid>spring-boot-loader</artifactid>
 <scope>provided</scope>
</dependency>

3.2.jarlauncher说明

jarlauncher作为引导类 ,当调用java -jar 命令时,将调用 main 方法,实际上调用的是 **「jarlauncher#launch」**方法,该方法继承于 org.springframework.boot.loader.launcher

简化层次关系为:

三分钟带你了解SpringBoot真正的启动引导类

jarlauncher#launch代码如下

protected void launch(string[] args) throws exception {
  jarfile.registerurlprotocolhandler();
  classloader classloader = createclassloader(getclasspatharchives());
  launch(args, getmainclass(), classloader);
}

「聚句解析」

「1,.jarfile.registerurlprotocolhandler();」

spring boot 生成的 fat jar,在被 java -jar 引导时,其内部的 jar 文件无法被 sun.net.www.protocol.jar.handler 处理。微信公众号搜索, [java学习之道] ,回复 ‘福利' 2t 资料等你来拿~

所以 springboot 实现了,org.springframework.boot.loader.jar.handler

jarfile.registerurlprotocolhandler(), 就注册 org.springframework.boot.loader.jar.handler

「2.classloader classloader = createclassloader(getclasspatharchives());」

创建classloader。

getclasspatharchives 核心判断是 isnestedarchive 方法。

isnestedarchive 被 jarlauncher 覆写了。其实现如下:

static final string boot_inf_classes = "boot-inf/classes/";

static final string boot_inf_lib = "boot-inf/lib/";
@override
protected boolean isnestedarchive(archive.entry entry) {
  if (entry.isdirectory()) {
   return entry.getname().equals(boot_inf_classes);
  }
  return entry.getname().startswith(boot_inf_lib);
}

也就是说,只要 **「满足以boot-inf/classes/和boot-inf/lib/都是classloader加载」**的范围。

解压的jar,查看也与只对应

三分钟带你了解SpringBoot真正的启动引导类

3. launch(args, getmainclass(), classloader);

protected void launch(string[] args, string mainclass, 
        classloader classloader)
   throws exception {
  thread.currentthread().setcontextclassloader(classloader);
  createmainmethodrunner(mainclass, args, classloader).run();
}

查看 createmainmethodrunner 的 run 方法,如下:

public class mainmethodrunner {
  // 省略部分代码
  public void run() throws exception {
  class<?> mainclass = thread.currentthread().getcontextclassloader()
     .loadclass(this.mainclassname);
    method mainmethod = 
      mainclass.getdeclaredmethod("main", string[].class);
    mainmethod.invoke(null, new object[] { this.args });
  }
}

其中 mainclass,来自 /meta-inf/manifest.mf 中的 start-class 属性。

「即,jarlauncher 是同进程内,通过反射调用 start-class 对应类,即 xxxapplication 的 main 方法。」

4.总结

springboot 项目的实际启动类是 org.springframework.boot.loader.jarlauncher

「在 jarlauncher 内部通过反射调用 xxapplication 类的 main 方法。具体实现位于 mainmethodrunner中。」

到此这篇关于三分钟带你了解springboot真正的启动引导类的文章就介绍到这了,更多相关springboot 启动引导类内容请搜索以前的文章或继续浏览下面的相关文章希望大家以后多多支持!