我已经用spring boot创建了一个项目。我有hikariConfig可以使用autocommmit属性设置为false来创建连接池的数据源。使用在@Transaction注释的DataSourceTransactionManager方法内部运行jdbcTemplate进行批处理插入。在程序执行后,我看不到数据被插入到Db中。如果我在hikariconfig中将autocommit设置为true,则可以正常工作。
@SpringBootApplication
public class Application {
public static void main(String[] args) {
SpringApplication.run(Application.class, args);
}
}
@Component
@EnableTransactionManagement
public class DataSourceConfig {
@Bean (name = "dateSourceForSqlServer")
public DataSource dataSourceForSqlServer () {
HikariConfig hikariConfig = new HikariConfig();
hikariConfig.setConnectionTimeout(10000L);
hikariConfig.setIdleTimeout(10000L);
hikariConfig.setMinimumIdle(1);
hikariConfig.setMaximumPoolSize(1);
hikariConfig.setMaxLifetime(600000L);
hikariConfig.setConnectionTestQuery("select 1");
hikariConfig.setValidationTimeout(4000L);
hikariConfig.setJdbcUrl("jdbc:sqlserver://localhost:1433;database=invt_mgmt");
hikariConfig.setUsername("sa");
hikariConfig.setPassword("sql_server_pass_123");
hikariConfig.setDriverClassName("com.microsoft.sqlserver.jdbc.SQLServerDriver");
hikariConfig.setAutoCommit(false);
return new HikariDataSource(hikariConfig);
}
@Bean (name = "jdbcTemplateForSqlServer")
public JdbcTemplate jdbcTemplateForSqlServer () {
JdbcTemplate jdbcTemplate = new JdbcTemplate();
jdbcTemplate.setDataSource(dataSourceForSqlServer());
return jdbcTemplate;
}
@Primary
@Bean(name = "invtMgmtTxMangerForSqlServer")
public DataSourceTransactionManager transactionManager() {
DataSourceTransactionManager manager = new DataSourceTransactionManager();
manager.setDataSource(dataSourceForSqlServer());
return manager;
}
}
@Component
public class startBean {
@Autowired
private Business Business;
@PostConstruct
public void startApp() throws SQLException {
Business.insertContainerHierarchy();
Business.insertContainerHierarchy();
}
}
@Component
class public Business {
@Autowired
@Qualifier("jdbcTemplateForSqlServer")
private JdbcTemplate jdbcTemplateForSqlServer;
String insertIntStudent = "INSERT INTO student (id, name) Values(?, ?)";
@Transactional(value = "invtMgmtTxMangerForSqlServer")
public void insertContainerHierarchy () throws SQLException {
System.out.println(TransactionSynchronizationManager.isActualTransactionActive());
System.out.println(TransactionSynchronizationManager.getCurrentTransactionName());
Date start = new Date();
for (int i = 0; i < 500; i++) {
System.out.println(i);
jdbcTemplateForSqlServer.batchUpdate(insertIntStudent, new BatchPreparedStatementSetter() {
@Override
public void setValues(PreparedStatement ps, int i) throws SQLException {
ps.setInt(1, i);
ps.setString(2, String.valueOf(i));
}
@Override
public int getBatchSize() {
return 1000;
}
});
}
System.out.println(new Date().getTime() - start.getTime());
}
}
我使用了TransactionSynchronizationManager.isActualTransactionActive()
,它在执行该方法时重现为真。
Q1。为什么不插入数据?执行该方法后,Transaction应该自动提交吗?
Q2。如果使用Spring事务,数据库连接自动提交值会有所不同吗?
Q3。当前如何在自动提交设置为true的情况下插入?
答案 0 :(得分:0)
您正在尝试从@PostConstruct
方法中调用事务包装的代理。对于该bean,所有初始化都可以完成,但对于上下文的其余部分则不一定。并不是所有的代理都可以在那个时候设置。
我建议实现ApplicationListener<ContextRefreshedEvent>
接口,以触发该类内部的任何数据创建。这将确保在设置完整个上下文之后将调用它:
@Component
public class OnContextInitialized implements
ApplicationListener<ContextRefreshedEvent> {
@Autowired
private Business Business;
@Override public void onApplicationEvent(ContextRefreshedEvent event) {
Business.insertContainerHierarchy();
Business.insertContainerHierarchy();
}
}