Eclipse中已终止的Spring Boot应用程序 - 未调用Shutdown挂钩

时间:2016-08-28 17:12:53

标签: java eclipse spring spring-boot redis

我有一个Spring Boot + Spring Data Redis / KeyValue项目。我设置了一个Spring配置文件来运行嵌入了所有依赖项的应用程序。所以在启动时,我启动了一个嵌入式Redis服务器。当我在Eclipse中启动时,一切正常,除了我希望在停止Spring Boot应用程序时停止Redis服务器。所以我设置了几个关闭钩子,但是当我从Eclipse终止应用程序时它们没有被调用。

他们是关于SO的类似问题,我创建了这个问题,希望有一个Redis解决方案。此外,这些类似的问题都不是Spring Boot特有的。

我尝试了很多东西:

  • Spring Boot的ExitCodeGenerator;
  • DisposableBean;
  • @PreDestroy;
  • 我尝试了ShutdownHook(在@ mp911de的回答之后)

没有人被召唤。

也许有一个Redis选项,超时,保持活着......我不知道的是什么? 如何确保在我的Spring Boot应用程序突然停止时停止Redis服务器

=>我看到了这个Embedded Redis for spring boot,但是在意外杀死应用程序时没有调用@PreDestroy

以下是一些类似的问题:

我还在eclipse.org上看到这篇文章,讨论在从eclipse停止应用程序时如何调用shutdown hook:Graceful shutdown of Java Applications

以下是我的所有相关代码:

在启动时启动嵌入式 Redis服务器的组件(我试图阻止它!):

@Component
public class EmbeddedRedis implements ExitCodeGenerator, DisposableBean{

    @Value("${spring.redis.port}")
    private int redisPort;

    private RedisServer redisServer;

    @PostConstruct
    public void startRedis() throws IOException {
        redisServer = new RedisServer(redisPort);
        redisServer.stop();
        redisServer.start();
    }

    @PreDestroy
    public void stopRedis() {
        redisServer.stop();
    }

    @Override
    public int getExitCode() {
        redisServer.stop();
        return 0;
    }

    @Override
    public void destroy() throws Exception {
        redisServer.stop();
    }
}

application.properties:

spring.redis.port=6379

Spring Boot App:

@SpringBootApplication
@EnableRedisRepositories
public class Launcher {

    public static void main(String[] args){
        new SpringApplicationBuilder() //
        .sources(Launcher.class)//
        .run(args);
    }

    @Bean
    public RedisTemplate<String, Model> redisTemplate() {
        RedisTemplate<String, Model> redisTemplate = new RedisTemplate<String, Model>();
        redisTemplate.setConnectionFactory(jedisConnectionFactory());
        return redisTemplate;
    }


    @Bean
    public JedisConnectionFactory jedisConnectionFactory() {
        JedisConnectionFactory jedisConnectionFactory = new JedisConnectionFactory();
        jedisConnectionFactory.setHostName("localhost");
        return jedisConnectionFactory;
    }
}

Redis服务器(〜嵌入式)运行:

$ ps -aef | grep redis
 ... /var/folders/qg/../T/1472402658070-0/redis-server-2.8.19.app *:6379  

嵌入式Redis maven依赖项:

<dependency>
    <groupId>com.github.kstyrc</groupId>
    <artifactId>embedded-redis</artifactId>
    <version>0.6</version>
</dependency>

5 个答案:

答案 0 :(得分:9)

使用SpringBoot和Eclipse时,可以安装STS(Spring Tool Suite)eclipse插件来实现正常关机。

安装完成后,将应用程序作为“Spring Boot App”运行,而不是常规的“Java应用程序”(运行/调试配置)

确保选中“启用生命周期管理”复选框复选框,当您单击红色方块按钮停止应用程序时,它将执行正常关闭而不是硬杀死。

编辑:

值得注意的是,有两个“红色方形按钮”。一个在“启动”工具栏中,一个在“控制台”面板中。启动工具栏中的那个仍然执行硬杀,但控制台中的那个允许正常关闭spring-boot应用程序(使用STS启动)

答案 1 :(得分:5)

经过调查,结果发现Spring-Boot只是终止了应用程序而没有任何机会运行关闭钩子。

由于System.exit(1);的连线方式,我找到了解决方法/黑客攻击。执行main方法时,tomcat在另一个线程中启动,main方法继续执行直到完成。这样,当您点击“Enter&#39;”时,我就可以插入终止逻辑

应用程序正常启动,控制台只需等待输入键执行@SpringBootApplication @EnableRedisRepositories public class Launcher { public static void main(String[] args){ new SpringApplicationBuilder() // .sources(Launcher.class)// .run(args); System.out.println("Press 'Enter' to terminate"); new Scanner(System.in).nextLine(); System.out.println("Exiting"); System.exit(1); } }

@PreDestroy

从Eclipse启动应用程序时,我现在点击进入控制台,而不是从界面终止应用程序。 现在,触发了挂钩({{1}})并且Redis服务器已停止!

不是我希望的,但至少目前嵌入式Redis服务器已停止使用该应用程序,我不必手动杀死它。

答案 2 :(得分:3)

ExitCodeGenerator要求应用程序调用{​​{1}}方法

exit

您可以另外注册一个shutdown hook。钩子将被称为

  • 当VM正常终止时(System.exit(SpringApplication .exit(SpringApplication.run(SampleBatchApplication.class, args))); )。
  • 响应中断(System.exitCtrl+C)或信号(SIGINTSIGHUP)。

在某些情况下,如果VM没有干净地关闭(SIGTERM,内部VM错误,本机方法中的错误),则无法保证是否调用了关闭挂钩。

SIGKILL组件的代码如下所示:

EmbeddedRedis

答案 3 :(得分:1)

我也遇到了同样的问题,尽管Redis却没有。我想要最大的可移植性,所以其他开发人员如果不想的话就不必添加STS。您可以将此添加到任何Spring Boot应用程序以提供干净关闭。这是我所做的,详细说明了OP的答案。

将此添加到您的主类或任何@Configuration类中

    @Bean
    public ApplicationRunner systemExitListener() {
        return args -> {
            if (args.getOptionValues("exitListener") != null) {
                System.out.println("Press Enter to exit application");
                new Scanner(System.in).nextLine();
                System.out.println("Exiting");
                System.exit(0);
            }
        };
    }

然后在Eclipse(或您选择的IDE中,甚至在命令行中)中,添加参数--exitListener以激活代码。在Eclipse中,该选项位于“运行配置”的“参数”选项卡上的“程序参数”框中。

答案 4 :(得分:0)

解决在停止 maven 构建并启用调试时杀死 Spring 启动应用程序的问题: Maven 构建 -> JRE -> VM 参数
添加 -Dspring-boot.run.fork=false

相关问题