it-roy-ru.com

Запуск кода после запуска Spring Boot

Я хочу запустить код после того, как мой spring-boot app начнет отслеживать каталог на предмет изменений. 

Я попытался запустить новый поток, но службы @Autowired не были установлены в этот момент.

Мне удалось найти ApplicationPreparedEvent, который срабатывает до установки аннотаций @Autowired. В идеале я хотел бы, чтобы событие запускалось, когда приложение готово для обработки http-запросов.

Есть ли лучшее событие для использования или лучший способ запуска кода после того, как приложение работает в spring-boot ?

118
stewsters

Пытаться:

@Configuration
@EnableAutoConfiguration
@ComponentScan
public class Application extends SpringBootServletInitializer {

    @SuppressWarnings("resource")
    public static void main(final String[] args) {
        ConfigurableApplicationContext context = SpringApplication.run(Application.class, args);

        context.getBean(Table.class).fillWithTestdata(); // <-- here
    }
}
79
Anton Bessonov

Это так просто, как это: 

@EventListener(ApplicationReadyEvent.class)
public void doSomethingAfterStartup() {
    System.out.println("hello world, I have just started up");
}

Проверено на версии 1.5.1.RELEASE

165
cahen

Вы пробовали ApplicationReadyEvent? 

@Component
public class ApplicationStartup 
implements ApplicationListener<ApplicationReadyEvent> {

  /**
   * This event is executed as late as conceivably possible to indicate that 
   * the application is ready to service requests.
   */
  @Override
  public void onApplicationEvent(final ApplicationReadyEvent event) {

    // here your code ...

    return;
  }
}

Код от: http://blog.netgloo.com/2014/11/13/run-code-at-spring-boot-startup/

Это то, что Documentation упоминает о событиях запуска:

...

События приложения отправляются в следующем порядке по мере запуска приложения:

ApplicationStartedEvent отправляется в начале прогона, но до любая обработка кроме регистрации слушателей и инициализаторов.

ApplicationEnvironmentPreparedEvent отправляется, когда среда, используемая в контексте, известна, но перед контекстом создано.

ApplicationPreparedEvent отправляется непосредственно перед началом обновления, но после загрузки определений компонента.

ApplicationReadyEvent отправляется после обновления, и все связанные обратные вызовы были обработаны, чтобы указать, что приложение готово к запросы на обслуживание.

ApplicationFailedEvent отправляется, если при запуске возникает исключение.

...

76
raspacorp

Почему бы просто не создать компонент, который запускает ваш монитор при инициализации, что-то вроде:

@Component
public class Monitor {
    @Autowired private SomeService service

    @PostConstruct
    public void init(){
        // start your monitoring in here
    }
}

метод init не будет вызываться до тех пор, пока для компонента не будет выполнена какая-либо автоматическая разводка.

76
cjstehno

«Spring Boot» - это использование CommandLineRunner. Просто добавьте бобы такого типа, и все готово. В Spring 4.1 (Boot 1.2) также есть SmartInitializingBean, который получает обратный вызов после того, как все инициализировано. И есть SmartLifecycle (с весны 3).

57
Dave Syer

Вы можете расширить класс с помощью ApplicationRunner , переопределить метод run() и добавить туда код. 

import org.springframework.boot.ApplicationRunner;

@Component
public class ServerInitializer implements ApplicationRunner {

    @Override
    public void run(ApplicationArguments applicationArguments) throws Exception {

        //code goes here

    }
}
29
Gimhani

ApplicationReadyEvent действительно полезен, только если задача, которую вы хотите выполнить, не является обязательной для правильной работы сервера. Хороший пример - запуск асинхронной задачи для отслеживания изменений.

Однако если ваш сервер находится в состоянии «не готов» до тех пор, пока задача не будет завершена, то лучше реализовать SmartInitializingSingleton, потому что вы получите обратный вызов до ваш порт REST был открыт, а ваш сервер Открыт для бизнеса.

Не поддавайтесь искушению использовать @PostConstruct для задач, которые должны выполняться только один раз. Вы получите грубый сюрприз, когда заметите, что он вызывается несколько раз ...

15
Andy Brown

С пружинной конфигурацией:

@Configuration
public class ProjectConfiguration {
    private static final Logger log = 
   LoggerFactory.getLogger(ProjectConfiguration.class);

   @EventListener(ApplicationReadyEvent.class)
   public void doSomethingAfterStartup() {
    log.info("hello world, I have just started up");
  }
}
15
freemanpolys

Используйте бин SmartInitializingSingleton весной> 4.1

@Bean
public SmartInitializingSingleton importProcessor() {
    return () -> {
        doStuff();
    };

}

В качестве альтернативы может быть реализован bean-компонент CommandLineRunner или аннотирование метода bean-компонента с помощью @PostConstruct.

9
Jeff

Предоставление примера для ответа Дейва Сайера, который работал как обаяние:

@Component
public class CommandLineAppStartupRunner implements CommandLineRunner {
    private static final Logger logger = LoggerFactory.getLogger(CommandLineAppStartupRunner.class);

    @Override
    public void run(String...args) throws Exception {
        logger.info("Application started with command-line arguments: {} . \n To kill this application, press Ctrl + C.", Arrays.toString(args));
    }
}
3
Paulo Pedroso

просто внедрите CommandLineRunner для приложения весенней загрузки . Вам нужно реализовать метод run,

public classs SpringBootApplication implements CommandLineRunner{

    @Override
        public void run(String... arg0) throws Exception {
        // write your logic here 

        }
}
1
prasad kp

Попробуйте это, и он запустит ваш код, когда контекст приложения полностью запустится.

 @Component
public class OnStartServer implements ApplicationListener<ContextRefreshedEvent> {

    @Override
    public void onApplicationEvent(ContextRefreshedEvent arg0) {
                // EXECUTE YOUR CODE HERE 
    }
}
1
Kalifornium

Мне очень нравится предложение использовать аннотацию EventListener @cahen ( https://stackoverflow.com/a/44923402/9122660 ), поскольку она очень чистая. К сожалению, я не смог заставить это работать в настройках Spring + Kotlin. Что работает для Kotlin, так это добавление класса в качестве параметра метода:

@EventListener 
fun doSomethingAfterStartup(event: ApplicationReadyEvent) {
    System.out.println("hello world, I have just started up");
}
0
Ostecke