SpringApplication 初始化

2024-06-23 21:38

本文主要是介绍SpringApplication 初始化,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

写在前面:在调试代码的时候,遇到小的细节都会记录,由于技术水平不高,请多留言指正。

  • Step Into 指跳入,Step Over 指下一步
    复杂情况一般指定行数:Step Into(11)跳入第11行,Step Over (33-35)从33行走到35行。
  • // 中为手动添加注释,纯英文的为自带注释
    注:一般为此代码中的疑难点 重点解释

SpringApplication初始化

一、 @SpringBootApplication

  • 开启debug模式
    在这里插入图片描述
  • Step into
	* @param source【当前的启动类的class】  the source to load* @param args 【main方法对应的参数,此时为空数组】the application arguments (usually passed from a Java main method)public static ConfigurableApplicationContext run(Object source, String... args) {return run(new Object[]{source}, args);}
  • Step Into
public static ConfigurableApplicationContext run(Object[] sources, String[] args) {//【初始化SpringApplication,调用其run方法,这里是以构造函数的方式,初始化了其他的参数】return new SpringApplication(sources).run(args);}

:对final的理解:new SpringApplication(source):通过构造函数的方式传了一个source参数给SpringApplilcation:

// 这里的source时final变量 定义如下private final Set<Object> sources = new LinkedHashSet<Object>();
-->
public SpringApplication(Object... sources) {initialize(sources);}private void initialize(Object[] sources) {if (sources != null && sources.length > 0) {this.sources.addAll(Arrays.asList(sources));}}final的值类型不可变,引用类型,引用的指向不可变,但被指向的对象可变这里final指向new LinkedHashSet,在【this.sources】拿到地址的时候,将对象赋值了。但不可以 进行:source==xxx(xxx为其他对象,这种操作)
ide会提示错误

在这里插入图片描述

  • Step Into
    SpringApplication.java
public ConfigurableApplicationContext run(String... args) {// 1.初始化stopWatcch(计数器),调用其start方法开始计时StopWatch stopWatch = new StopWatch();stopWatch.start();ConfigurableApplicationContext context = null;FailureAnalyzers analyzers = null;// 2.设置系统属性java.awt.headless,这里设置为true,表示运行在服务器端,在没有显示器和鼠标键盘的模式下工作,模拟输入输出设备功能configureHeadlessProperty();// 3.调用springApplicationRunListeners#startingSpringApplicationRunListeners listeners = getRunListeners(args);listeners.starting();try {// 4.创建一个defaultApplicationArguments对象,他持有args参数(main函数中传入),调用prepareEnvironment方法ApplicationArguments applicationArguments = new DefaultApplicationArguments(args);ConfigurableEnvironment environment = prepareEnvironment(listeners, applicationArguments);// 5.打印bannerBanner printedBanner = printBanner(environment);// 6.创建springboot上下文context = createApplicationContext();// 7.初始化FailureAnalyzersanalyzers = new FailureAnalyzers(context);// 8.调用prepareContextprepareContext(context, environment, listeners, applicationArguments, printedBanner);// 9.调用AbstractApplicationContext#refreshrefreshContext(context);// 10.在容器完成刷新后,依次调用注册的RunnersafterRefresh(context, applicationArguments);// 11.调用SpringApplicationRunListeners#finishedlisteners.finished(context, null);// 12.停止计时stopWatch.stop();if (this.logStartupInfo) {new StartupInfoLogger(this.mainApplicationClass).logStarted(getApplicationLog(), stopWatch);}return context;} catch (Throwable ex) {// 13.初始化过程中出现异常时调用handelRunFailure进行处理,抛出IllegalStateExceptionhandleRunFailure(context, listeners, analyzers, ex);throw new IllegalStateException(ex);}
}

:以上代码及注释会在github

  • 1 github本地上传
  • 2 github工具:左侧的目录树 Tree

第一步:初始化stopWatcch(计数器),调用其start方法开始计时
	StopWatch stopWatch = new StopWatch();stopWatch.start();

StopWatch.class

/*** Construct a new stop watch with the given id.* Does not start any task.* @param id identifier for this stop watch.* Handy when we have output from multiple stop watches* and need to distinguish between them.*/public StopWatch(String id) {this.id = id;}/*** Return the id of this stop watch, as specified on construction.* @return the id (empty String by default)* @since 4.2.2* @see #StopWatch(String)*/public String getId() {return this.id;}/*** Determine whether the TaskInfo array is built over time. Set this to* "false" when using a StopWatch for millions of intervals, or the task* info structure will consume excessive memory. Default is "true".*/public void setKeepTaskList(boolean keepTaskList) {this.keepTaskList = keepTaskList;}/*** Start an unnamed task. The results are undefined if {@link #stop()}* or timing methods are called without invoking this method.* @see #stop()*/public void start() throws IllegalStateException {start("");}/*** Start a named task. The results are undefined if {@link #stop()}* or timing methods are called without invoking this method.* @param taskName the name of the task to start* @see #stop()*/public void start(String taskName) throws IllegalStateException {if (this.running) {throw new IllegalStateException("Can't start StopWatch: it's already running");}this.running = true;this.currentTaskName = taskName;this.startTimeMillis = System.currentTimeMillis();}/*** Stop the current task. The results are undefined if timing* methods are called without invoking at least one pair* {@code start()} / {@code stop()} methods.* @see #start()*/public void stop() throws IllegalStateException {if (!this.running) {throw new IllegalStateException("Can't stop StopWatch: it's not running");}long lastTime = System.currentTimeMillis() - this.startTimeMillis;this.totalTimeMillis += lastTime;this.lastTaskInfo = new TaskInfo(this.currentTaskName, lastTime);if (this.keepTaskList) {//默认为truethis.taskList.add(lastTaskInfo);}++this.taskCount;this.running = false;this.currentTaskName = null;}

:用哪种方式表示当前时间:

方式一:this.startTimeMillis = System.currentTimeMillis();
方式二:this.startTimeMills = new Date().getTime()

推荐博文:System.currentTimeMillis()和new Date().getTime()区别

// 下面是第二种方式的实现源码:public Date() {this(System.currentTimeMillis());}public Date(long date) {fastTime = date;}public long getTime() {return getTimeImpl();}private final long getTimeImpl() {if (cdate != null && !cdate.isNormalized()) {normalize();}return fastTime;}
可见:第二种方式内部也是采用System.currentTimeMills()方式。建议使用第一种方式(同源码)
第十三步:此方法的stop()

此行代码:

		this.lastTaskInfo = new TaskInfo(this.currentTaskName, lastTime);if (this.keepTaskList) {//默认为truethis.taskList.add(lastTaskInfo);}

:TaskInfo:

    public static final class TaskInfo {private final String taskName;private final long timeMillis;TaskInfo(String taskName, long timeMillis) {this.taskName = taskName;this.timeMillis = timeMillis;}/*** Return the name of this task.*/public String getTaskName() {return this.taskName;}/*** Return the time in milliseconds this task took.*/public long getTimeMillis() {return this.timeMillis;}/*** Return the time in seconds this task took.*/public double getTimeSeconds() {return (this.timeMillis / 1000.0);}}
  1. final变量的初始化:
    若为对象变量(非静态)可在定义时赋值,或者在构造函数中赋值(上述代码中就是构造函数中赋值;若为类变量(静态)则可在定义时或静态代码块中赋值)推荐博文 final变量的初始化方式
  2. 类的命名规则:
    本类为StopWatch.class的静态内部类
    所有内部类会在编译的时候产生相对应的class文件,非匿名内部类类名规则为 OutClass$InnerClass (外部类类名与内部类类名中间用 $ 连接) 匿名内部类类名则为OutClass$数字(OutClass$1,OutClass$2,OutClass$3)—摘录自匿名内部类类名规则($1,$2)
    如图:ide中的类名
    在这里插入图片描述
  • 单例模式下的静态内部类
public class StaticInnerClassMode {private StaticInnerClassMode(){};public static StaticInnerClassMode getInstance(){return InnerClass.staticInnerClassMode;}private static class InnerClass {private static StaticInnerClassMode staticInnerClassMode =new StaticInnerClassMode();}public static void main(String[] args) {System.err.println(StaticInnerClassMode.getInstance());// System.err.println(StaticInnerClassMode.getInstance());}
}

推荐博文:java静态内部类以及单例应用

第二步:设置系统属性java.awt.headless

这里设置为true,表示运行在服务器端,在没有显示器和鼠标键盘的模式下工作,模拟输入输出设备功能 configureHeadlessProperty();

  • step into

    这里设置了系统参数:

	private void configureHeadlessProperty() {System.setProperty(SYSTEM_PROPERTY_JAVA_AWT_HEADLESS,System.getProperty(SYSTEM_PROPERTY_JAVA_AWT_HEADLESS, Boolean.toString(this.headless)));}
  • step into #setProperty
public static String setProperty(String key, String value) {checkKey(key);SecurityManager sm = getSecurityManager();if (sm != null) {sm.checkPermission(new PropertyPermission(key,SecurityConstants.PROPERTY_WRITE_ACTION));}return (String) props.setProperty(key, value);}

这里是 setProperty方法的实现,我们来看下
SecurityManager sm = getSecurityManager();这句如何返回一个Security对象

  • step into #getSecurityManager();
   private static volatile SecurityManager security = null;public static SecurityManager getSecurityManager() {return security;}

注:volatile关键字在这里出现,单例模式中有个双重校验if的写法,但是并不能完全保证线程安全,加上volatile 关键字可以有效避免(指令重排) 推荐博文:单例模式为什么要用Volatile关键字

第三步:设置系统属性java.awt.headless

这篇关于SpringApplication 初始化的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/1088350

相关文章

JVM 的类初始化机制

前言 当你在 Java 程序中new对象时,有没有考虑过 JVM 是如何把静态的字节码(byte code)转化为运行时对象的呢,这个问题看似简单,但清楚的同学相信也不会太多,这篇文章首先介绍 JVM 类初始化的机制,然后给出几个易出错的实例来分析,帮助大家更好理解这个知识点。 JVM 将字节码转化为运行时对象分为三个阶段,分别是:loading 、Linking、initialization

c++的初始化列表与const成员

初始化列表与const成员 const成员 使用const修饰的类、结构、联合的成员变量,在类对象创建完成前一定要初始化。 不能在构造函数中初始化const成员,因为执行构造函数时,类对象已经创建完成,只有类对象创建完成才能调用成员函数,构造函数虽然特殊但也是成员函数。 在定义const成员时进行初始化,该语法只有在C11语法标准下才支持。 初始化列表 在构造函数小括号后面,主要用于给

@postconstruct初始化的操作

从Java EE 5规范开始,Servlet中增加了两个影响Servlet生命周期的注解(Annotion);@PostConstruct和@PreDestroy。这两个注解被用来修饰一个非静态的void()方法 。写法有如下两种方式: @PostConstruct Public void someMethod() {}

spring和tomcat初始化的类和注解

1.InitializingBean接口为bean提供了初始化方法的方式,它只包括afterPropertiesSet方法,凡是继承该接口的类,在初始化bean的时候会执行该方法。 spring为bean提供了两种初始化bean的方式,实现InitializingBean接口,实现afterPropertiesSet方法,或者在配置文件中同过init-method指定,两种方式可以同时使用 实

LabVIEW环境中等待FPGA模块初始化完成

这个程序使用的是LabVIEW环境中的FPGA模块和I/O模块初始化功能,主要实现等待FAM(Field-Programmable Gate Array Module,FPGA模块)的初始化完成,并处理初始化过程中的错误。让我们逐步分析各部分的功能: 1. Wait for FAM Initialization框架 此程序框架用于等待I/O模块成功初始化。如果在5秒钟内模块没有完成配

dp(背包问题) 恰好、至少、至多初始化

状态表示的初始化(一般情况) f[i][j] i:前i件物品 体积至少为j 枚举体积时可以是负数(体积为负数时等价于体积为0) max f[i][j] = {-0x3f} f[i][0] = 0min f[i][j] = { 0x3f} f[i][0] = 0cnt f[0][0] = 1 体积至多为j 枚举体积时不能是负数 max f[i][j] = 0min f[i][j]

Windows11上使用WSL2,提示:系统尚未使用systemd作为初始化系统(PID 1)启动

前言 略 报错信息 System has not been booted with systemd as init system (PID 1). Can't operate. Failed to connect to bus: Host is down 解决方法 使用如下命令 # windows终端,执行如下命令wsl --update# 登录ubuntu系统,执行如下命令s

WapApp初始化样式表

@charset "utf-8";/* 样式初始化--------------------------------------------------*/body {margin: 0;padding: 0;background-color: #f6f6f6;font: normal 67.5% Helvetica, Arial, sans-serif;min-width: 240px;}*

Spark Core源码精读计划7 | Spark执行环境的初始化

推荐阅读 《Spark源码精度计划 | SparkConf》 《Spark Core源码精读计划 | SparkContext组件初始化》 《Spark Core源码精读计划3 | SparkContext辅助属性及后初始化》 《Spark Core源码精读计划4 | SparkContext提供的其他功能》 《Spark Core源码精读计划5 | 事件总线及ListenerBus》 《Spa

Spark Core源码精读计划3 | SparkContext辅助属性及后初始化

推荐阅读 《关于MQ面试的几件小事 | 消息队列的用途、优缺点、技术选型》         《关于MQ面试的几件小事 | 如何保证消息队列高可用和幂等》 《关于MQ面试的几件小事 | 如何保证消息不丢失》 《关于MQ面试的几件小事 | 如何保证消息按顺序执行》 《关于MQ面试的几件小事 | 消息积压在消息队列里怎么办》 《关于Redis的几件小事 | 使用目的与问题及线程模型》 《关于Red