«1. Обзор

Управление жизненным циклом приложения Spring Boot очень важно для готовой к производству системы. Контейнер Spring обрабатывает создание, инициализацию и уничтожение всех компонентов Bean с помощью ApplicationContext.

Особое внимание в этой статье уделяется фазе уничтожения жизненного цикла. В частности, мы рассмотрим различные способы закрытия приложения Spring Boot.

Чтобы узнать больше о том, как настроить проект с помощью Spring Boot, ознакомьтесь со статьей Spring Boot Starter или ознакомьтесь с конфигурацией Spring Boot.

2. Конечная точка завершения работы

По умолчанию в приложении Spring Boot включены все конечные точки, кроме /shutdown; это, естественно, часть конечных точек актуатора.

Вот зависимость Maven для их настройки:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-actuator</artifactId>
</dependency>

И, если мы хотим также настроить поддержку безопасности, нам нужно:

<dependency>
    <groupId>org.springframework.boot</groupId>
    <artifactId>spring-boot-starter-security</artifactId>
</dependency>

Наконец, мы включаем конечную точку выключения в приложении. файл свойств:

management.endpoints.web.exposure.include=*
management.endpoint.shutdown.enabled=true
endpoints.shutdown.enabled=true

Обратите внимание, что мы также должны предоставить доступ к любым конечным точкам привода, которые мы хотим использовать. В приведенном выше примере мы предоставили все конечные точки привода, включая конечную точку /shutdown.

Чтобы закрыть приложение Spring Boot, мы просто вызываем метод POST следующим образом:

curl -X POST localhost:port/actuator/shutdown

В этом вызове порт представляет собой порт привода.

3. Закрытие контекста приложения

Мы также можем вызвать метод close() напрямую, используя контекст приложения.

Давайте начнем с примера создания контекста и его закрытия:

ConfigurableApplicationContext ctx = new 
  SpringApplicationBuilder(Application.class).web(WebApplicationType.NONE).run();
System.out.println("Spring Boot application started");
ctx.getBean(TerminateBean.class);
ctx.close();

Это уничтожает все компоненты, снимает блокировки, затем закрывает фабрику компонентов. Чтобы проверить завершение работы приложения, мы используем стандартный обратный вызов жизненного цикла Spring с аннотацией @PreDestroy:

public class TerminateBean {

    @PreDestroy
    public void onDestroy() throws Exception {
        System.out.println("Spring Container is destroyed!");
    }
}

Мы также должны добавить bean-компонент этого типа:

@Configuration
public class ShutdownConfig {

    @Bean
    public TerminateBean getTerminateBean() {
        return new TerminateBean();
    }
}

Вот результат после запуска этого примера: ~ ~~

Spring Boot application started
Closing [email protected]
DefaultLifecycleProcessor - Stopping beans in phase 0
Unregistering JMX-exposed beans on shutdown
Spring Container is destroyed!

Здесь важно помнить: при закрытии контекста приложения родительский контекст не затрагивается из-за отдельных жизненных циклов.

3.1. Закрытие текущего контекста приложения

В приведенном выше примере мы создали дочерний контекст приложения, а затем использовали метод close() для его уничтожения.

Если мы хотим закрыть текущий контекст, одно из решений — просто вызвать конечную точку привода /shutdown.

Однако мы также можем создать свою собственную конечную точку:

@RestController
public class ShutdownController implements ApplicationContextAware {
    
    private ApplicationContext context;
    
    @PostMapping("/shutdownContext")
    public void shutdownContext() {
        ((ConfigurableApplicationContext) context).close();
    }

    @Override
    public void setApplicationContext(ApplicationContext ctx) throws BeansException {
        this.context = ctx;
        
    }
}

Здесь мы добавили контроллер, который реализует интерфейс ApplicationContextAware и переопределяет метод установки для получения текущего контекста приложения. Затем в методе сопоставления мы просто вызываем метод close().

Затем мы можем вызвать нашу новую конечную точку, чтобы отключить текущий контекст:

curl -X POST localhost:port/shutdownContext

Конечно, если вы добавите подобную конечную точку в реальное приложение, вы также захотите защитить ее.

4. Выход из SpringApplication

SpringApplication регистрирует ловушку завершения работы с JVM, чтобы убедиться, что приложение завершает работу надлежащим образом.


Бины могут реализовать интерфейс ExitCodeGenerator для возврата определенного кода ошибки:

ConfigurableApplicationContext ctx = new SpringApplicationBuilder(Application.class)
  .web(WebApplicationType.NONE).run();

int exitCode = SpringApplication.exit(ctx, new ExitCodeGenerator() {
@Override
public int getExitCode() {
        // return the error code
        return 0;
    }
});

System.exit(exitCode);

Тот же код с применением лямбда-выражений Java 8:

SpringApplication.exit(ctx, () -> 0);

После вызова System.exit(exitCode) программа завершается с кодом возврата 0:

Process finished with exit code 0

5. Завершить процесс приложения

Наконец, мы также можем закрыть приложение Spring Boot извне приложения с помощью сценария bash. Наш первый шаг для этой опции — заставить контекст приложения записать свой PID в файл:

SpringApplicationBuilder app = new SpringApplicationBuilder(Application.class)
  .web(WebApplicationType.NONE);
app.build().addListeners(new ApplicationPidFileWriter("./bin/shutdown.pid"));
app.run();

Затем создайте файл shutdown.bat со следующим содержимым:

kill $(cat ./bin/shutdown.pid)

Выполнение shutdown.bat извлекает идентификатор процесса из файла shutdown.pid и использует команду kill для завершения приложения загрузки.

6. Заключение

В этом кратком обзоре мы рассмотрели несколько простых методов, которые можно использовать для закрытия работающего приложения Spring Boot.

Хотя выбор подходящего метода остается за разработчиком; все эти методы должны использоваться по замыслу и по назначению.

«Например, .exit() предпочтительнее, когда нам нужно передать код ошибки в другую среду, скажем, JVM для дальнейших действий. Использование PID приложения дает больше гибкости, так как мы также можем запускать или перезапускать приложение с помощью скрипта bash.

Наконец, параметр /shutdown позволяет завершать работу приложений извне через HTTP. Во всех остальных случаях .close() будет работать отлично.

Как обычно, полный код для этой статьи доступен в проекте GitHub.