java - Spring数据库事务未在Db中提交数据

标签 java sql-server spring spring-boot spring-transactions

我用 spring boot 创建了一个项目。我有 hikariConfig 来创建连接池的数据源,并将 autocommmit 属性设置为 false。使用在 DataSourceTransactionManager 的 @Transaction 注释的方法内运行的 jdbcTemplate 进行批量插入。我无法看到程序执行后数据插入到 Db 中。如果我在 hikariconfig 中将自动提交设置为 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() ,它在执行该方法时返回 true。

Q1。为什么数据没有被插入,事务应该在方法执行后自动提交?

第二季度。如果使用 Spring 事务,数据库连接自动提交值会产生任何影响吗?

第三季度。目前我如何能够在自动提交设置为 true 的情况下插入?

最佳答案

您正尝试从 @PostConstruct 内调用事务包装的代理。方法。对于该 bean,所有初始化可能都已完成,但对于上下文的其余部分则不一定。并非所有代理都可以在此时设置。

我建议实现ApplicationListener<ContextRefreshedEvent>接口(interface)以触发该类内的任何数据创建。这将确保在设置整个上下文后调用它:

@Component
public class OnContextInitialized implements
  ApplicationListener<ContextRefreshedEvent> {

    @Autowired
    private Business Business;

    @Override public void onApplicationEvent(ContextRefreshedEvent event) {
        Business.insertContainerHierarchy();
        Business.insertContainerHierarchy();
    }
}

关于java - Spring数据库事务未在Db中提交数据,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/55889089/

相关文章:

java - Spring :用 <list/> 进行干燥。将所有值从列表复制到列表

java - 我如何在子类java中使用父类(super class)方法

java - MethodArgumentNotValidException 的 validator 仅处理相同类型的约束

java - 询问用户是否愿意继续

sql - 如何在查询过程中清除 SQL 查询的结果消息数据?

java - Springboot 2.0无法访问SQL Server数据库

sql-server - 将 bacpac 文件还原到 SQL Server 2014 会导致错误 SQL72014

java - Spring中entityManager上的NullPointerException

java - 如何使用线程每次进行检查(例如在后台运行)直到我的条件为真

java - 如何为 grails 编写 XLSX 自定义渲染器