通过在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 ),它在“连接被拒绝:连接”时失败,可能是因为服务器没有运行。谢谢你。
这个解决方案对我有用。如果应用程序作为 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;
}