先说具体的步骤。
还是一样,先导入jar包,在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>
第二,使用配置文件的话,需要配置事务通知,这部分Spring给我们做了的,我我们只需要做一些简单的配置就可以了。(以什么开头的方法,注意name里的 * ,也可以直接使用 * 表示所有的方法)
<!--配置事务通知-->
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<tx:attributes>
<!--配置哪些方法使用什么样的事务,配置事务的传播特性-->
<tx:method name="add*" propagation="REQUIRED"/>
<tx:method name="delete*" propagation="REQUIRED"/>
<tx:method name="update*" propagation="REQUIRED"/>
<tx:method name="get*" read-only="true"/>
<tx:method name="*" propagation="REQUIRED"/>
</tx:attributes>
</tx:advice>
第三,就是配置AOP织入事务。(所以这里也可以看出,它做事务管理的原理是使用的AOP)
<!--配置aop织入事务-->
<aop:config>
<aop:pointcut id="txPointcut" expression="execution(* com.huchuan.service.*.*(..))"/>
<aop:advisor advice-ref="txAdvice" pointcut-ref="txPointcut"/>
</aop:config>
第四,配置事务通知哪里可以配置很多属性,用来设置连接超时(timeout 秒),事务的传播行为(propagation 七种),事务的隔离级别(四种 在高并发场景可能产生脏读,不可重复读,虚读),是否只读(readOnly),回滚(rollbackFor),不回滚(noRollbackFor)。
最后,我遇到了一个可以说是坑吧(实际上还是自己的知识面不够广)。
问题是我使用xml配置文件来做的声明式事务管理,代码没错,配置也没问题。但是一个事务是这样的,在Service层的方法配置了事务,然后我的一个方法里先是增加一个用户,修改一个用户,删除一个用户(我把delete写成deletes,这样当执行到这个地方就会抛异常,程序结束),最后返回用户列表。
可是在运行的时候,虽然在删除的那个地方抛了异常,程序结束,可是增加和修改却在数据库中生效了,程序是没有问题的。问题在于我用的mysql数据库它创建表默认的引擎是MyISAM,这个引擎是不支持事务的,把用户表的引擎改为INNODB就可以了,这个引擎支持事务。