我试图以编程方式重新启动我的Spring Application,而无需用户干预。
基本上,我有一个页面可以切换应用程序的模式(实际上意味着切换当前活动的配置文件),据我所知,我必须重新启动上下文。
目前,我的代码非常简单,仅用于重新启动(顺便说一下,这就是Kotlin):
context.close() application.setEnvironment(context.environment) ClassUtils.overrideThreadContextClassLoader(application.javaClass.classLoader) context = application.run(*argsArray)
但是,当我执行context.close()JVM 的那一刻立即存在。我也曾尝试过,context.refresh()但这似乎只是杀死了Tomcat / Jetty(以防万一这是Tomcat问题,尝试了两者),然后什么也没发生。
context.close()
context.refresh()
我也看到以编程方式重新启动SpringBoot应用程序,但是从这些答案来看,似乎没有任何帮助。此外,我查看了据称具有/restart端点的Spring Actuator,但似乎不再存在了?
/restart
即使Alex的解决方案 有效 ,我也不相信仅包含2个依赖项(Actuator和Cloud Context)就能够执行一项操作。相反,我结合了他的答案并修改了我的代码,以执行我想要的操作。
Actuator
Cloud Context
因此,首先,使用和执行代码 至关重要 。我有以下处理重启的端点方法:new Thread()``setDaemon(false);
new Thread()``setDaemon(false);
val restartThread = Thread { logger.info("Restarting...") Thread.sleep(1000) SpringMain.restartToMode(AppMode.valueOf(change.newMode.toUpperCase())) logger.info("Restarting... Done.") } restartThread.isDaemon = false restartThread.start()
该Thread.sleep(1000)不是必需的,但我希望我的控制器输出实际上重新启动应用程序前视图。
Thread.sleep(1000)
SpringMain.restartToMode 具有以下内容:
SpringMain.restartToMode
@Synchronized fun restartToMode(mode: AppMode) { requireNotNull(context) requireNotNull(application) // internal logic to potentially produce a new arguments array // close previous context context.close() // and build new one using the new mode val builder = SpringApplicationBuilder(SpringMain::class.java) application = builder.application() context = builder.build().run(*argsArray) }
凡context与application来自main在启动应用程序的方法:
context
application
main
val args = ArrayList<String>() lateinit var context: ConfigurableApplicationContext lateinit var application: SpringApplication @Throws(Exception::class) @JvmStatic fun main(args: Array<String>) { this.args += args val builder = SpringApplicationBuilder(SpringMain::class.java) application = builder.application() context = builder.build().run(*args) }
我不确定这是否会引起任何问题。如果有,我将更新此答案。希望这对其他人有帮助。