一,原始JDBC的事务操作
无论编程式事务还是声明式事务,底层都封装原生JDBC对事务的操作
jdbc的所有类在java.sql包中,操作Jdbc通过DriveManager获取Connection连接对象,通过这个连接对象进行各种操作
jdbc操作事务的方法:
- commit():提交事务
- rollback():回滚事务
- setAutoCommit(boolean autoCommit):设置连接是否自动提交事务(默认为true自动提交事务)
- getAutoCommit():获取当前连接的事务是否为自动提交
- setTransactionIsolation(int leven):设置事务的隔离基本
- getTransactionIsolation:获取当前的隔离级别
DBC提供了5中事务隔离级别,它们以常量的形式在Connection接口中定义,除了TRANSACTION_NONE外,其它4种隔离级别 与 MySQL的事务隔离级别一样
- TRANSACTION_NONE(不支持事务)
- TRANSACTION_READ_UNCOMMITTED
- TRANSACTION_READ_COMMITTED
- TRANSACTION_REPEATABLE_READ
- TRANSACTION_SERIALIZABLE
例:使用原生JDBC事务测试:
public class JDBC_Transaction {
public static void main(String[] args) {
Connection conn = null;
PreparedStatement pstmt1 = null;
PreparedStatement pstmt2 = null;
try {
//1.获取连接
conn = JDBCUtils.getConnection();
//开启事务(手动提交事务)
conn.setAutoCommit(false);
//2.定义sql
//2.1 张三 - 500
String sql1 = "update account set balance = balance - ? where id = ?";
//2.2 李四 + 500
String sql2 = "update account set balance = balance + ? where id = ?";
//3.获取执行sql对象
pstmt1 = conn.prepareStatement(sql1);
pstmt2 = conn.prepareStatement(sql2);
//4. 设置参数
pstmt1.setDouble(1,500);
pstmt1.setInt(2,1);
pstmt2.setDouble(1,500);
pstmt2.setInt(2,2);
//5.执行sql pstmt1.executeUpdate();
// 手动制造异常
int i = 3/0;
pstmt2.executeUpdate();
//提交事务 conn.commit();
} catch (Exception e) {
//事务回滚
try {
if(conn != null) {
conn.rollback();
}
} catch (SQLException e1) {
e1.printStackTrace();
}
e.printStackTrace();
}finally {
JDBCUtils.close(pstmt1,conn);
JDBCUtils.close(pstmt2,null);
}
二,编程式事务(无案例)
编程式事务是声明式事务的基础,编程式事务就是通过编写事务代码实现事务的控制,而声明式事务则通过Spring配置事务代码简化代码编写,实现配置约定大于代码编写
编程式事务控制有几个重要对象:
- PlatformTransactionManager接口:spring的事务管理器,它里面提供了我们常用的操作事务的方法
- TransactionDefinition:事务的定义信息对象
- TransactionStatus接口:提供事务的具体运行状态信息
一,PlatformTransactionManager
spring的事务管理器,它里面提供了我们常用的操作事务的方法
| 方法 | 说明 |
| TransactionStatus getTransaction(TransactionDefination defination) | 获取事务的状态信息 |
| void commit(TransactionStatus status) | 提交事务 |
| void rollback(TransactionStatus status) | 回滚事务 |
| …… | …… |
注:PlatformTransactionManger是接口类型,不同的Dao层技术有不同的实现方式,例:JDBC或Mybatis是org.springframework.jdbc.DatasourceTransactionManager来实现此接口,而Hibernate使用org.springframework.orm.hibernate5.HibernateTransactionManager实现,但它们的方法都大同小异
二, TransactionDefinition
事务的定义信息对象,封装事务的相关参数如果隔离级别的设置和获取,事务的传播行为设置和获取
|
方法
|
说明
|
|
int getIsolationLevel()
|
获取事务的隔离级别
|
|
int getPropogationBehavior()
|
获取事务的传播行为 |
|
int getTimeout()
|
获取超时时间
|
|
boolean isReadOnly()
|
是否只读
|
|
……
|
……
|
- ISOLATION_DEFAULT:使用数据库的默认隔离级别
- ISOLATION_READ_UNCOMMITTED:读未提交,一个事务可以读取另一个事务未提交的数据
- ISOLATION_READ_COMMITTED:读已提交,一个事务只能读取另一个事务已经提交的数据,可以解决脏读问题
- ISOLATION_REPEATABLE_READ:不可重复读,一个事务在整个过程中可以多次重复执行某个查询,并返回的记录都相同,可以避免脏读和不可重复读
- ISOLATION_SERIALIZABLE:串行化,一次只能一个用户执行操作(锁),可以解决幻读问题,但效率太低
事务的传播行为:解决业务方法调用业务方法时事务的同一性问题
- REQUIRED:如果当前没有事务,就新建一个事务,如果已经有一个事务,就加入这个事务(默认)
- SUPPORTS:支持当前事务,如果当前没有事务,就以非事务的方式执行(没有事务)
- MANDATORY:使用当前的事务,如果当前没有事务,就抛出异常
- REQUERS_NEW:新建事务,如果当前已在事务中,就把事务挂起
- NOT_SUPPORTED:以非事务的方式执行操作,如果当前存在事务,就把当前事务挂起
- NEVER:以非事务方式运行,如果当前存在事务,就抛出异常
- BESTED:如果当前存在事务,则在嵌套事务内执行,如果当前没有事务,则执行REQUIRED类似操作
- 超时时间:默认为1,没有超时限制
- 是否只读:建议查询时设置
三,TransactionStatus
设置和获取事务的状态信息
|
方法
|
说明
|
|
boolean hasSavepoint()
|
是否存储回滚点
|
|
boolean isCompleted()
|
事务是否完成
|
|
boolean isNewTransaction()
|
是否是新事务
|
|
boolean isRollbactOnly()
|
事务是否回滚
|
|
……
|
……
|
三,声明式事务
一,声明式事务的概念
Spring的声明式事务就是采用声明式的方式处理事务,声明在在spring的xml配置文件中进行声明,提供spring配置文件声明的方式代替编程式事务,简化操作
声明式事务的作用:
- 事务管理不侵入开发模块,事务管理本质上是系统层面上的,开发的业务代码不应该和系统代码进行耦合,声明式事务就对两者进行了松耦合
- 事务管理和维护方便,在需要修改事务时,只需要修改配置,而不需要修改代码
声明式事务底层使用了AOP,通过切面,在程序运行的过程中对业务进行控制
二,基于XML的事务控制
maven依赖:
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-tx</artifactId>
<version>5.3.0</version>
</dependency>
命名空间:
xmlns:tx="http://www.springframework.org/schema/tx" xsi:schemaLocation="http://www.springframework.org/schema/tx https://www.springframework.org/schema/tx/spring-tx.xsd"
配置事务分为三步:
- 配置事务管理器
- 配置事务的通知
- 将事务做为切面通过AOP织入具体业务方法中
一,配置事务管理器
配置事务管理器通过org.springframework.jdbc.datasource.DataSourceTransactionManager类,这个类需要加入DataSource实现
例:
<!--值从properties文件中获取的-->
<bean id="datasource" class="com.alibaba.druid.pool.DruidDataSource">
<property name="username" value="${user}"/>
<property name="password" value="${password}"/>
<property name="url" value="${url}"/>
<property name="driverClassName" value="${drive}"/>
</bean>
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<!--通过set映入DataSource-->
<property name="dataSource" ref="datasource"/>
</bean>
二,配置事务通知
在xml配置文件中通过<tx:adrice>标签配置事务通知
例:
<!--
transaction-manager:引入事务管理器
<tx:attributes>:配置具体通知
-->
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="*" isolation="READ_UNCOMMITTED" timeout="100" propagation="REQUIRED"/>
</tx:attributes>
</tx:advice>
事务通知的属性信息设置:声明式事务是简化了编程式事务,在编程式事务中的重要类的配置参数也可以用于声明中( <tx:method>中设置)
- name:切点的名称,*代表所有
- isolation:设置事务的管理级别
- propogation:事务的传播级别
- timeout:事务的超时时间
- read-only:是否只读
三,事务的AOP织入
将事务做为切面织入业务方法中
在传统的AOP使用时通过<aop:aspect>来配置切面,但配置事务时,AOP通过了专门的标签<aop:advisor>来进行配置
例:
<!--
<aop:config>:AOP配置
<aop:adviso>:专门用于配置事务
pointcut:切点表达式,将事务织那个具体方法中
-->
<aop:config>
<aop:advisor advice-ref="txAdvice" pointcut="execution(* com.service..*(..))"/>
</aop:config>
三,基于注解的声明式事务控制
注解配置声明式事务的步骤:
- 在xml文件中开启包扫描
- 配置事务管理器(TransactionManager)
- 开启注解事务驱动
- 注解实现事务控制
1,前置配置
<!--开启包扫描-->
<context:component-scan base-package="com"/>
<!--配置事务管理器-->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="datasource"/>
</bean>
<!--开启注解事务驱动:transaction-manager加入事务管理器-->
<tx:annotation-driven transaction-manager="transactionManager"/>
2,注解控制
通过@Transactional()配置事务,这个注解替代了xml文件中的通知配置和AOP切点织入
- 标注注解的方法就为切点
- 通知配置可以在参数中进行配置
例:
@Transactional(isolation = Isolation.READ_COMMITTED)
public void accounts(String inname, String outname, int money){
da.in(inname,money);
System.out.println(1/0);
da.out(outname,money);
}
四,dome
一,xml配置
① dao
public class deal implements deal_interface {
@Autowired
JdbcTemplate jdbcTemplate;
//转入
public int in(String name,int money) {
String SQL = "update jdbctemplatetest set money=money+? where name=?";
int i = jdbcTemplate.update(SQL,money,name);
return i;
}
//转出
public int out(String name,int money) {
String SQL = "update jdbctemplatetest set money=money-? where name=?";
int i = jdbcTemplate.update(SQL,money,name);
return i;
}
}
② Service
public class tx_service {
private deal da;
public tx_service(deal de) {
this.da=de;
}
public void accounts(String inname, String outname, int money){
da.in(inname,money);
da.out(outname,money);
}
}
③ xml配置
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:tx="http://www.springframework.org/schema/tx"
xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop https://www.springframework.org/schema/aop/spring-aop.xsd
http://www.springframework.org/schema/tx https://www.springframework.org/schema/tx/spring-tx.xsd">
<!--引入外部mysql连接properties的信息文件-->
<context:property-placeholder location="jdbc.properties"/>
<!--配置DataSource,使用的是druid实现datasource类-->
<bean id="datasource" class="com.alibaba.druid.pool.DruidDataSource">
<property name="username" value="${user}"/>
<property name="password" value="${password}"/>
<property name="url" value="${url}"/>
<property name="driverClassName" value="${drive}"/>
</bean>
<!--配置JdbcTemplate-->
<bean id="jdbctemplate" class="org.springframework.jdbc.core.JdbcTemplate">
<property name="dataSource" ref="datasource"/>
</bean>
<!--dao中的deal类,实现转入和转出-->
<bean id="deal" class="com.dao.deal"/>
<!--业务类-->
<bean id="tx_service" class="com.service.tx_service">
<constructor-arg name="de" ref="deal"/>
</bean>
<!--transaction管理器-->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="datasource"/>
</bean>
<!--通知配置-->
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="*" isolation="READ_UNCOMMITTED" timeout="100" propagation="REQUIRED"/>
</tx:attributes>
</tx:advice>
<!--将事务做为切面进行织入-->
<aop:config>
<aop:advisor advice-ref="txAdvice" pointcut="execution(* com.service..*(..))"/>
</aop:config>
</beans>
④ 测试
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = {"classpath:/transaction.xml"})
public class test {
@Autowired
tx_service tx;
@Test
public void test(){
tx.accounts("FQ","WQL",1000);
}
}
二,注解配置
① dao (添加一个@Repository注解)
@Repository
public class deal implements deal_interface {
@Autowired
JdbcTemplate jdbcTemplate;
//转入
public int in(String name,int money) {
String SQL = "update jdbctemplatetest set money=money+? where name=?";
int i = jdbcTemplate.update(SQL,money,name);
return i;
}
//转出
public int out(String name,int money) {
String SQL = "update jdbctemplatetest set money=money-? where name=?";
int i = jdbcTemplate.update(SQL,money,name);
return i;
}
}
② Service
@Service
public class tx_ann_service {
@Autowired
deal da;
public void accounts(String inname, String outname, int money){
da.in(inname,money);
System.out.println(1/0);
da.out(outname,money);
}
}
③ xml配置
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:context="http://www.springframework.org/schema/context"
xmlns:tx="http://www.springframework.org/schema/tx"
xmlns:aop="http://www.springframework.org/schema/aop"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context https://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop https://www.springframework.org/schema/aop/spring-aop.xsd
http://www.springframework.org/schema/tx https://www.springframework.org/schema/tx/spring-tx.xsd">
<!--引入外部mysql连接properties的信息文件-->
<context:property-placeholder location="jdbc.properties"/>
<!--配置DataSource,使用的是druid实现datasource类-->
<bean id="datasource" class="com.alibaba.druid.pool.DruidDataSource">
<property name="username" value="${user}"/>
<property name="password" value="${password}"/>
<property name="url" value="${url}"/>
<property name="driverClassName" value="${drive}"/>
</bean>
<!--配置JdbcTemplate-->
<bean id="jdbctemplate" class="org.springframework.jdbc.core.JdbcTemplate">
<property name="dataSource" ref="datasource"/>
</bean>
<!--transaction管理器-->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="datasource"/>
</bean>
<!--transaction注解事务驱动:需要添加事务管理器transaction-manager-->
<tx:annotation-driven transaction-manager="transactionManager"/>
<!--包扫描-->
<context:component-scan base-package="com"/>
</beans>
④ 测试
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration(locations = {"classpath:/transaction_ann.xml"})
public class test {
@Autowired
tx_service tx;
@Test
public void test(){
tx.accounts("FQ","WQL",1000);
}
}





Comments | NOTHING