使用Spring和MyBatis抛出缺少@Transactional的异常

时间:2017-01-20 16:20:58

标签: spring mybatis spring-mybatis

我正在尝试设置使用Spring和MyBatis的网络应用。

以下是重要的代码片段。

pom.xml中的Maven依赖项:

    ...
    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-core</artifactId>
        <version>4.3.5.RELEASE</version>
    </dependency>

    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-web</artifactId>
        <version>4.3.5.RELEASE</version>
    </dependency>

    <dependency>
        <groupId>org.springframework</groupId>
        <artifactId>spring-jdbc</artifactId>
        <version>4.3.5.RELEASE</version>
    </dependency>

    <dependency>
        <groupId>org.postgresql</groupId>
        <artifactId>postgresql</artifactId>
        <version>9.4.1212.jre7</version>
    </dependency>

    <dependency>
        <groupId>org.mybatis</groupId>
        <artifactId>mybatis</artifactId>
        <version>3.4.2</version>
    </dependency>

    <dependency>
        <groupId>org.mybatis</groupId>
        <artifactId>mybatis-spring</artifactId>
        <version>1.3.1</version>
    </dependency>
    ...

Spring bean的配置:

@Configuration
@EnableTransactionManagement
public class DatabaseConfiguration {

    @Value("classpath:db/mybatis/mybatis-configuration.xml")
    private Resource myBatisConfiguration;

    @Bean
    public DataSource dataSource() {
        final DriverManagerDataSource dataSource = new DriverManagerDataSource();
        dataSource.setDriverClassName("org.postgresql.Driver");
        dataSource.setUrl("jdbc:postgresql://127.0.0.1:5432/ehdb");
        dataSource.setUsername(/*my username*/);
        dataSource.setPassword(/*my password*/);
        return dataSource;
    }

    @Bean
    public PlatformTransactionManager transactionManager() {
        final DataSourceTransactionManager transactionManager = new DataSourceTransactionManager(dataSource());
        transactionManager.setValidateExistingTransaction(true);
        return transactionManager;
    }

    @Bean
    public SqlSessionFactoryBean sqlSessionFactoryBean() {
        final SqlSessionFactoryBean sqlSessionFactory = new SqlSessionFactoryBean();
        sqlSessionFactory.setDataSource(dataSource());
        sqlSessionFactory.setConfigLocation(myBatisConfiguration);
        return sqlSessionFactory;
    }

    @Bean
    public SqlSession sqlSession() throws Exception {
        return new SqlSessionTemplate(sqlSessionFactoryBean().getObject());
    }
}

这是一个应该在事务方法中调用一些MyBatis语句的服务:

@Service
public class HelloManagerImpl implements HelloManager {
    private final HelloDao helloDao;

    public HelloManagerImpl(@Autowired final HelloDao helloDao) {
        this.helloDao = helloDao;
    }

    @Override
    @Transactional
    public String doSomething() {
        helloDao.insertRow(); // a row is inserted into DB table via MyBatis; bean sqlSession is autowired in HelloDao
        throw new RuntimeException(); // transaction will be rolled back here
    }
}

如果我调用方法doSomething,它会按预期工作。数据库表中不会出现新行,因为抛出RuntimeException会导致事务回滚。

如果我注释掉throw语句并重复实验,数据库表中会出现一个新行。这是预期的行为。

现在,如果我另外注释掉@Transactional注释并调用doSomething(),则该方法会成功,并在表中插入一个新行。如果没有事务存在,MyBatis似乎会自动为INSERT语句创建一个事务。

我希望在最后一种情况下失败。如果我忘记写@Transactional注释,那可能是个错误。如果在这种情况下抛出异常迫使我修复我的代码而不是静默创建一些事务,那就没问题了。

有没有办法实现这个?

感谢您的帮助。

2 个答案:

答案 0 :(得分:1)

我建议将所有事务设置为只读以实现最后一种情况,并且只注释应该在数据库中写入的方法。例如,您的服务将如下所示:

@Service
@Transactional(readOnly = true)
public class HelloManagerImpl implements HelloManager {
  private final HelloDao helloDao;

  public HelloManagerImpl(@Autowired final HelloDao helloDao) {
    this.helloDao = helloDao;
  }

  @Override
  @Transactional(readOnly = false)
  public String doSomething() {
    helloDao.insertRow(); // a row is inserted into DB table via MyBatis; bean sqlSession is autowired in HelloDao
    throw new RuntimeException(); // transaction will be rolled back here
  }
}

答案 1 :(得分:0)

事务的行为由Spring本身管理,而Spring在RuntimeException的情况下会对事务进行反转。

如果您没有提供@Transactional注释,那么Spring不会将其视为一个事务,并且在RuntimeException的情况下不会执行任何操作。因此,将@Transactional注释放在服务方法之上是一个好习惯。