温馨提示×

温馨提示×

您好,登录后才能下订单哦!

密码登录×
登录注册×
其他方式登录
点击 登录注册 即表示同意《亿速云用户服务条款》

深入浅析SpringBoot的启动引导类

发布时间:2020-11-04 14:44:20 来源:亿速云 阅读:210 作者:Leah 栏目:开发技术

今天就跟大家聊聊有关深入浅析SpringBoot的启动引导类,可能很多人都不太了解,为了让大家更加了解,小编给大家总结了以下内容,希望大家根据这篇文章可以有所收获。

引言

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<&#63;> 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 方法。

看完上述内容,你们对深入浅析SpringBoot的启动引导类有进一步的了解吗?如果还想了解更多知识或者相关内容,请关注亿速云行业资讯频道,感谢大家的支持。

向AI问一下细节

免责声明:本站发布的内容(图片、视频和文字)以原创、转载和分享为主,文章观点不代表本网站立场,如果涉及侵权请联系站长邮箱:is@yisu.com进行举报,并提供相关证据,一经查实,将立刻删除涉嫌侵权内容。

AI