提问者:小点点

如何在Spring启动应用程序启动时启动H2 TCP服务器?


通过在SpringBootServletInitializer主方法中添加以下行,我可以在作为Spring Boot应用程序运行应用程序时启动H2 TCP服务器(文件中的数据库):

@SpringBootApplication
public class NatiaApplication extends SpringBootServletInitializer {
    public static void main(String[] args) {
        Server.createTcpServer().start();
        SpringApplication.run(NatiaApplication.class, args);
    }
}

但是如果我在Tomcat上运行WAR文件,它就不起作用了,因为main方法没有被调用。在beans被初始化之前,有没有更好的通用方法在应用程序启动时启动H2 TCP服务器?我使用Flyway (autoconfig ),它在“连接被拒绝:连接”时失败,可能是因为服务器没有运行。谢谢你。


共3个答案

匿名用户

这个解决方案对我有用。如果应用程序作为 Spring Boot 应用程序运行,并且它还在 Tomcat 上运行,它将启动 H2 服务器。将 H2 服务器创建为 Bean 不起作用,因为 Flyway Bean 是较早创建的,并且在“连接被拒绝”时失败。

@SpringBootApplication
@Log
public class NatiaApplication extends SpringBootServletInitializer {

    public static void main(String[] args) {
        startH2Server();
        SpringApplication.run(NatiaApplication.class, args);
    }

    @Override
    protected SpringApplicationBuilder configure(SpringApplicationBuilder application) {
        startH2Server();
        return application.sources(NatiaApplication.class);
    }

    private static void startH2Server() {
        try {
            Server h2Server = Server.createTcpServer().start();
            if (h2Server.isRunning(true)) {
                log.info("H2 server was started and is running.");
            } else {
                throw new RuntimeException("Could not start H2 server.");
            }
        } catch (SQLException e) {
            throw new RuntimeException("Failed to start H2 server: ", e);
        }
    }
}

匿名用户

是的,直接从留档开始,您可以使用bean引用:

<bean id = "org.h2.tools.Server"
        class="org.h2.tools.Server"
        factory-method="createTcpServer"
        init-method="start"
        destroy-method="stop">
<constructor-arg value="-tcp,-tcpAllowOthers,-tcpPort,8043" />

还有一个servlet侦听器选项可以自动启动/停止它。

这回答了你的问题,但我认为如果它与你的Spring Boot应用程序一起部署,你可能应该使用嵌入式模式。这样速度更快,占用的资源也更少。您只需指定正确的URL,数据库就会启动:

jdbc:h2:/usr/share/myDbFolder

(直接从备忘单中取出)。

匿名用户

有一个在其他答案中没有考虑到的警告。您需要注意的是,启动服务器是对< code>DataSource bean的暂时依赖。这是因为< code>DataSource只需要一个网络连接,而不需要bean关系。

这导致的问题是spring-boot在创建DataSource之前不知道需要启动h2数据库,因此您最终可能会在应用程序启动时出现连接异常。

使用Spring框架,这不是问题,因为您将DB服务器启动放在根配置中,将数据库作为子配置。使用Spring boot AFAIK只有一个上下文。

要解决这个问题,您可以创建一个<code>可选

@Bean(destroyMethod = "close")
public DataSource dataSource(Optional<Server> h2Server) throws PropertyVetoException {
    HikariDataSource ds = new HikariDataSource();
    ds.setDriverClassName(env.getProperty("db.driver"));
    ds.setJdbcUrl(env.getProperty("db.url"));
    ds.setUsername(env.getProperty("db.user"));
    ds.setPassword(env.getProperty("db.pass"));
    return ds;
}