Spring bean上的System.exit()销毁有效吗?

时间:2018-06-20 22:11:09

标签: java spring spring-boot

最近,我在使用Spring Boot(1.4.7)时遇到了完全关机问题。简而言之:

此lib通过自动配置注册bean。该bean出于某些未记录的未记录原因触发System.exit调用。当我尝试关闭Spring上下文时,这会导致死锁。可以使用以下代码重现此问题:

@Configuration
@EnableAutoConfiguration
public class App {
    public static void main(String[] args) throws InterruptedException {
        SpringApplication application = new SpringApplication(App.class);
        ConfigurableApplicationContext context = application.run(args);
        Thread.sleep(1000);
        context.close();
    }

    @Bean
    public Object someThirdPartyBeanIcantChange() {
        return new Object() {
            @PreDestroy
            public void tooSmartMethod() {
                //trigger some methods that will lead to
                System.exit(0);
            }
        };
    }
}

和Maven的POM内容:

<project xmlns="http://maven.apache.org/POM/4.0.0"
         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
         xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd">
    <modelVersion>4.0.0</modelVersion>

    <groupId>TEST</groupId>
    <artifactId>context-test</artifactId>
    <version>1.0-SNAPSHOT</version>

    <properties>
        <java.version>1.8</java.version>
    </properties>

    <parent>
        <groupId>org.springframework.boot</groupId>
        <artifactId>spring-boot-starter-parent</artifactId>
        <version>1.4.7.RELEASE</version>
    </parent>

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

    <build>
        <plugins>
            <plugin>
                <groupId>org.springframework.boot</groupId>
                <artifactId>spring-boot-maven-plugin</artifactId>
            </plugin>
        </plugins>
    </build>
</project>

据我了解:

context.close()导致调用带有注释的@PreDestroy方法。 这导致System.exit(0)导致默认的spring的shutdown钩子自身关闭上下文,该上下文本身已在主线程已经获取的内部监视器上阻塞,从而导致死锁。

所以问题是:这是在destroy方法中调用System.exit的有效行为吗?

还有什么想法,如果有必要使用弹簧关闭钩子(或其他类似的钩子),我该如何克服这种死锁(当然,除了改变bean的行为外)?

0 个答案:

没有答案