SpringCloud启动中SpringApplication构造方法执行多次

前言

最近在看源码的时候发现一个比较疑惑的地方就就是我启动spring boot会出现springApplication的构造方法和run方法多次调用。因此决定研究一下

正文

以下是我的启动代码

@SpringBootApplication
public class TestApplication {
    public static void main(String[] args) {
                SpringApplication springApplication = new SpringApplication(TestApplication .class);
                ConfigurableApplicationContext context = springApplication.run(args);
                context.close();
    }
}

经过debug发现以下流程

1 第一次调用

1.1 调用构造方法
1.2 执行run方法中的listeners.starting()时 再次调用构造方法

2 第二次调用

2.1 调用构造方法
2.2 执行run方法中的ConfigurableEnvironment environment = this.prepareEnvironment(listeners, applicationArguments)时; 再次调用构造方法

3 第三次调用

3.1 调用构造方法和run方法(完成后返回之前的run方法继续执行)
3.2 启动容器

探究

根据粗略的跟踪源码发现以上三个流程让我们仔细深入进去探究一下

RestartApplicationListener

由1.2可知第一次跳转是执行listeners.starting()时进行的
可能是因为启动RestartApplicationListener监听并传播ApplicationStartingEvent事件时创建了Restarter对象并重重启。
关键代码如下首先是RestartApplicationListener的onApplicationStartingEvent方法处理spring容器启动事件

 private void onApplicationStartingEvent(ApplicationStartingEvent event) {
        String enabled = System.getProperty("spring.devtools.restart.enabled");
        if (enabled != null && !Boolean.parseBoolean(enabled)) {
            Restarter.disable();
        } else {
            String[] args = event.getArgs();
            DefaultRestartInitializer initializer = new DefaultRestartInitializer();
            boolean restartOnInitialize = !AgentReloader.isActive();
            Restarter.initialize(args, false, initializer, restartOnInitialize);
        }

    }

以上通过 Restarter.initialize(args, false, initializer, restartOnInitialize);这行代码进行Restarter初始化启动的代码,让我们接着往下

public static void initialize(String[] args, boolean forceReferenceCleanup, RestartInitializer initializer, boolean restartOnInitialize) {
        Restarter localInstance = null;
        Object var5 = INSTANCE_MONITOR;
        synchronized(INSTANCE_MONITOR) {
            if (instance == null) {
               //进行 Restarter的初始化操作
                localInstance = new Restarter(Thread.currentThread(), args, forceReferenceCleanup, initializer);
                instance = localInstance;
            }
        }

        if (localInstance != null) {
            localInstance.initialize(restartOnInitialize);
        }

    }

以上代码主要进行了Restarter的初始化操作并且通过 localInstance.initialize(restartOnInitialize);来进行重启

protected void initialize(boolean restartOnInitialize) {
        this.preInitializeLeakyClasses();
        if (this.initialUrls != null) {
            this.urls.addAll(Arrays.asList(this.initialUrls));
            if (restartOnInitialize) {
                this.logger.debug("Immediately restarting application");
                this.immediateRestart();
            }
        }

    }

 private void immediateRestart() {
        try {
            this.getLeakSafeThread().callAndWait(() -> {
                this.start(FailureHandler.NONE);
                this.cleanupCaches();
                return null;
            });
        } catch (Exception var2) {
            this.logger.warn("Unable to initialize restarter", var2);
        }

        SilentExitExceptionHandler.exitCurrentThread();
    }

以上代码关键的地方位判断是否是restart初始化 如果是则需要重启 并且调用LeakSafeThread进行重启
关键方法为this.start(FailureHandler.NONE); 让我们接着跟踪

    protected void start(FailureHandler failureHandler) throws Exception {
        Throwable error;
        do {
            error = this.doStart();
            if (error == null) {
                return;
            }
        } while(failureHandler.handle(error) != Outcome.ABORT);

    }

    private Throwable doStart() throws Exception {
        Assert.notNull(this.mainClassName, "Unable to find the main class to restart");
        URL[] urls = (URL[])this.urls.toArray(new URL[0]);
        ClassLoaderFiles updatedFiles = new ClassLoaderFiles(this.classLoaderFiles);
        ClassLoader classLoader = new RestartClassLoader(this.applicationClassLoader, urls, updatedFiles, this.logger);
        if (this.logger.isDebugEnabled()) {
            this.logger.debug("Starting application " + this.mainClassName + " with URLs " + Arrays.asList(urls));
        }

        return this.relaunch(classLoader);
    }

    protected Throwable relaunch(ClassLoader classLoader) throws Exception {
        RestartLauncher launcher = new RestartLauncher(classLoader, this.mainClassName, this.args, this.exceptionHandler);
        launcher.start();
        launcher.join();
        return launcher.getError();
    }

这里主要是启动RestartLauncher线程然后通过RestartLauncher线程的run方法进行重启 如下代码

public void run() {
        try {
            Class<?> mainClass = this.getContextClassLoader().loadClass(this.mainClassName);
            Method mainMethod = mainClass.getDeclaredMethod("main", String[].class);
            mainMethod.invoke((Object)null, this.args);
        } catch (Throwable var3) {
            this.error = var3;
            this.getUncaughtExceptionHandler().uncaughtException(this, var3);
        }

    }

如上RestartLauncher线程的run方法主要通过调用springboot启动类的main方法重新启动并刷新(因此此时已经把Restarter实例存入Restarter类中的instance中所以下次不会再初始化和重新构造了 因为没有重启jvm所以static中的数据没被刷新 所以存在)
以上为第一次调整并调用构造方法的原因和过程之后会再开章节详细讲解Restarter这个类的作用

2.2 BootstrapApplicationListener

第二次重启根据debug可以知道是通过调用run方法中的如下代码进行跳转的

this.prepareEnvironment(listeners, applicationArguments)

根据跟踪源码可以知道 这里不是重启,是在当前线程向 BootstrapApplicationListener发送一个ApplicationEnvironmentPreparedEvent广播,然后由BootstrapApplicationListener创建bootstrap上下文并返回

首先我们看BootstrapApplicationListener构造bootstrap的核心代码

        SpringApplicationBuilder builder = (new SpringApplicationBuilder(new Class[0])).profiles(environment.getActiveProfiles()).bannerMode(Mode.OFF).environment(bootstrapEnvironment).registerShutdownHook(false).logStartupInfo(false).web(WebApplicationType.NONE);
        if (environment.getPropertySources().contains("refreshArgs")) {
            builder.application().setListeners(this.filterListeners(builder.application().getListeners()));
        }
        省略代码
        builder.sources((Class[])sources.toArray(new Class[sources.size()]));
        ConfigurableApplicationContext context = builder.run(new String[0]);
        context.setId("bootstrap");

由上面源码可知在BootstrapApplicationListener中会调用构造方法和run方法进行创建bootstrap上下文 然后 将上下文返回,并继续执行原来的run方法来创建application上下文

©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 212,294评论 6 493
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 90,493评论 3 385
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 157,790评论 0 348
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 56,595评论 1 284
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 65,718评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 49,906评论 1 290
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,053评论 3 410
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 37,797评论 0 268
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,250评论 1 303
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 36,570评论 2 327
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 38,711评论 1 341
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,388评论 4 332
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,018评论 3 316
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 30,796评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,023评论 1 266
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 46,461评论 2 360
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 43,595评论 2 350

推荐阅读更多精彩内容

  • 数据结构 数据元素 是数据的基本单位,即数据集合中的个体。 一个数据元素可以由若干 数据项(Data Item) ...
    Cytosine阅读 292评论 0 1
  • 如果有一天我沉默了, 你会不会想到曾经的反驳。 如果有一天我崩溃了, 你会不会来了解一下。 如果有一天...
    忆往昔拾忆阅读 584评论 4 7