用spring来做声明式事务管理有两种方式,可以使用注解,也可以使用xml配置文件。它原理是使用AOP来生成一个代理对象来做的事务管理。先来说说步骤。
毫无疑问,先导入需要的包,和设置资源扫描路径(这个根据自己的资源文件的路径决定要不要加),在maven中。
<dependencies>
<dependency>
<groupId>mysql</groupId>
<artifactId>mysql-connector-java</artifactId>
<version>5.1.49</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-jdbc</artifactId>
<version>5.2.15.RELEASE</version>
</dependency>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-webmvc</artifactId>
<version>5.2.15.RELEASE</version>
</dependency>
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis</artifactId>
<version>3.5.7</version>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.13.2</version>
</dependency>
<dependency>
<groupId>org.aspectj</groupId>
<artifactId>aspectjweaver</artifactId>
<version>1.9.7</version>
</dependency>
<dependency>
<groupId>org.mybatis</groupId>
<artifactId>mybatis-spring</artifactId>
<version>2.0.6</version>
</dependency>
</dependencies>
<build>
<resources>
<resource>
<directory>src/main/resources</directory>
<includes>
<include>**/*.properties</include>
<include>**/*.xml</include>
</includes>
<filtering>true</filtering>
</resource>
<!--配置扫描java包下定义的资源文件,不这样做的话会导致java包下的资源文件不能被编译成class文件,导致用不了-->
<resource>
<directory>src/main/java</directory>
<includes>
<include>**/*.properties</include>
<include>**/*.xml</include>
</includes>
<filtering>true</filtering>
</resource>
</resources>
</build>
首先,不论哪种方式,都需要先在Spring配置文件中创建一个事务管理器。(Mybatis使用DataSourceTransactionManager,)
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource" />
</bean>
第二,使用注解的话,只需要开启注解就可以使用了。(它底层用的也是AOP,所以可以设置他的proxy-target-class="" 的值是否为true,即AOP的实现方式)
<tx:annotation-driven transaction-manager="transactionManager"/>
第三,使用注解@Transactional标识在类(方法上),这样就可以实现一个简单的事务管理功能了。一般标注在Service层。
第四,@Transactional里有很多属性,用来设置连接超时(timeout 秒),事务的传播行为(propagation 七种),事务的隔离级别(四种 在高并发场景可能产生脏读,不可重复读,虚读),是否只读(readOnly),回滚(rollbackFor),不回滚(noRollbackFor)。
最后,我遇到了一个可以说是坑吧(实际上还是自己的知识面不够广)。
问题是我使用注解来做的声明式事务管理,代码没错,配置也没问题。但是一个事务是这样的,在Service层加了注解@Transactional,然后我的一个方法里先是增加一个用户,修改一个用户,删除一个用户(我把delete写成deletes,这样当执行到这个地方就会抛异常,程序结束),最后返回用户列表。
可是在运行的时候,虽然在删除的那个地方抛了异常,程序结束,可是增加和修改却在数据库中生效了,程序是没有问题的。问题在于我用的mysql数据库它创建表默认的引擎是MyISAM,这个引擎是不支持事务的,把用户表的引擎改为INNODB就可以了,这个引擎支持事务。
我使用配置文件来做的声明式事务在下边这个连接。