事务
1.事务:指的是逻辑上一组操作,组成这个事务的各个执行单元,要么一起成功,要么一起失败!
2.事务的特性
- 原子性
- 一致性
- 隔离性
- 持久性
3.如果不考虑隔离性,引发安全性问题
- 读问题:
- 脏读: 事务A读取了事务B更新的数据,然后B回滚操作,那么A读取到的数据是脏数据
- 不可重复读::事务 A 多次读取同一数据,事务 B 在事务A多次读取的过程中,对数据作了更新并提交,导致事务A多次读取同一数据时,结果 不一致。
- 虚读: 系统管理员A将数据库中所有学生的成绩从具体分数改为ABCDE等级,但是系统管理员B就在这个时候插入了一条具体分数的记录,当系统管理员A改结束后发现还有一条记录没有改过来,就好像发生了幻觉一样,这就叫幻读。
- 写问题:
- 丢失更新:
- 小结:不可重复读的和幻读很容易混淆,不可重复读侧重于修改,幻读侧重于新增或删除。解决不可重复读的问题只需锁住满足条件的行,解决幻读需要锁表
4.如何解决安全性问题
-
读问题解决,设置数据库隔离级别
-
写问题解决可以使用 悲观锁和乐观锁的方式解决
-
通过设置事务的隔离级别
Read uncommitted: 读未提交,不能解决任何读取的问题。
Repeatable read: 可重复读,可解决脏读和不可重复读,但虚读、幻读可能会发生
Read committed: 读已提交,解决脏读,虚度/幻读 和 不可重复读有可能发生。
Serialzable : 解决所有读的问题。
Spring框架的事务管理相关的类和API
- PlatformTransactionManager接口 -- 平台事务管理器.(真正管理事务的类)。该接口有具体的实现类,根据不同的持久层框架,需要选择不同的实现类!
- TransactionDefinition接口 -- 事务定义信息.(事务的隔离级别,传播行为,超时,只读)
- TransactionStatus接口 -- 事务的状态
总结:上述对象之间的关系:平台事务管理器真正管理事务对象.根据事务定义的信息TransactionDefinition 进行事务管理,在管理事务中产生一些状态.将状态记录到TransactionStatus中
PlatformTransactionManager接口中实现类和常用的方法
-
接口的实现类
如果使用的Spring的JDBC模板或者MyBatis框架,需要选择DataSourceTransactionManager实现类
如果使用的是Hibernate的框架,需要选择HibernateTransactionManager实现类
-
该接口的常用方法
void commit(TransactionStatus status)
TransactionStatus getTransaction(TransactionDefinition definition)
void rollback(TransactionStatus status)
-
TransactionDefinition
事务隔离级别的常量
static int ISOLATION_DEFAULT -- 采用数据库的默认隔离级别
static int ISOLATION_READ_UNCOMMITTED
static int ISOLATION_READ_COMMITTED
static int ISOLATION_REPEATABLE_READ
static int ISOLATION_SERIALIZABLE
2.事务的传播行为常量(不用设置,使用默认值)
先解释什么是事务的传播行为:解决的是业务层之间的方法调用!!
PROPAGATION_REQUIRED(默认值) -- A中有事务,使用A中的事务.如果没有,B就会开启一个新的事务,将A包含进来.(保证A,B在同一个事务中),默认值!!
PROPAGATION_SUPPORTS -- A中有事务,使用A中的事务.如果A中没有事务.那么B也不使用事务. * PROPAGATION_MANDATORY -- A中有事务,使用A中的事务.如果A没有事务.抛出异常. * PROPAGATION_REQUIRES_NEW(记)-- A中有事务,将A中的事务挂起.B创建一个新的事务.(保证A,B没有在一个事务中) * PROPAGATION_NOT_SUPPORTED -- A中有事务,将A中的事务挂起.
PROPAGATION_NEVER -- A中有事务,抛出异常.
PROPAGATION_NESTED(记) -- 嵌套事务.当A执行之后,就会在这个位置设置一个保存点.如果B没有问题.执行通过.如果B出现异常,运行客户根据需求回滚(选择回滚到保存点或者是最初始状态)
事务的使用
银行转账功能
pom.xml配置
<?xml version="1.0" encoding="UTF-8"?> <project xmlns="http://maven.apache.org/POM/4.0.0" xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xsi:schemaLocation="http://maven.apache.org/POM/4.0.0 http://maven.apache.org/xsd/maven-4.0.0.xsd"> <modelVersion>4.0.0</modelVersion> <groupId>org.example</groupId> <artifactId>springT</artifactId> <version>1.0-SNAPSHOT</version> <packaging>war</packaging> <name>springT Maven Webapp</name> <properties> <project.build.sourceEncoding>UTF-8</project.build.sourceEncoding> <maven.compiler.source>1.8</maven.compiler.source> <maven.compiler.target>1.8</maven.compiler.target> <spring.version>5.2.5.RELEASE</spring.version> <aopalliance.version>1.0</aopalliance.version> <aspectj.weaver>1.9.2</aspectj.weaver> <!-- junit版本号 --> <junit.version>4.5</junit.version> </properties> <dependencies> <!-- spring --> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-core</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-beans</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-context</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-expression</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-tx</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-web</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-test</artifactId> <version>${spring.version}</version> </dependency> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.13</version> <scope>test</scope> </dependency> <!--切入点表达式的支持--> <dependency> <groupId> org.aspectj</groupId> <artifactId> aspectjweaver</artifactId> <version> 1.6.11</version> </dependency> <dependency> <groupId>junit</groupId> <artifactId>junit</artifactId> <version>4.12</version> <scope>compile</scope> </dependency> <!-- Spring-jdbc 用于配置JdbcTemplate --> <dependency> <groupId>org.springframework</groupId> <artifactId>spring-jdbc</artifactId> <version>5.2.5.RELEASE</version> </dependency> <!--c3p0--> <dependency> <groupId>com.mchange</groupId> <artifactId>c3p0</artifactId> <version>0.9.5.2</version> </dependency> <dependency> <groupId>mysql</groupId> <artifactId>mysql-connector-java</artifactId> <version>5.1.17</version> </dependency> <dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> <version>RELEASE</version> <scope>compile</scope> </dependency> <dependency> <groupId>org.projectlombok</groupId> <artifactId>lombok</artifactId> <version>RELEASE</version> <scope>compile</scope> </dependency> </dependencies> </project>
1.配置applicationContext.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/tx http://www.springframework.org/schema/tx/spring-tx.xsd http://www.springframework.org/schema/aop https://www.springframework.org/schema/aop/spring-aop.xsd"> <!--包的扫描--> <context:component-scan base-package="com.hp"/> <!--配置读取jdbc文件--> <context:property-placeholder location="classpath:db.properties"/> <!--c3p0配置--> <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource"> <property name="driverClass" value="${datasource.connection.driver_class}"/> <property name="jdbcUrl" value="${datasource.connection.url}"/> <property name="user" value="${datasource.connection.username}"/> <property name="password" value="${datasource.connection.password}"/> </bean> <!--构jdbcTemplate模板交给Spring容器管理--> <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate"> <!--引入数据源--> <property name="dataSource" ref="dataSource"/> </bean> <!--事务管理采用xml配置,代码中dao层和service层采用 注解方式--> <!--配置事务管理器--> <bean id="transaction" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> <property name="dataSource" ref="dataSource"/> </bean> <!-- 配置需要 事务的目标类(主要就是业务层的类),给目标类增强的代码配置--> <!-- <tx:advice id="txAdvice" transaction-manager="transaction">--> <!-- <tx:attributes>--> <!-- <tx:method name="transfer" propagation="REQUIRED"/>--> <!-- </tx:attributes>--> <!-- </tx:advice>--> <!--aop的配置--> <!-- <aop:config>--> <!-- <aop:pointcut id="pointcut1" expression="execution(* com.hp.service.DemoService.*(..))"/>--> <!-- <aop:advisor advice-ref="txAdvice" pointcut-ref="pointcut1"/>--> <!-- </aop:config>--> <!--开启注解事务--> <tx:annotation-driven transaction-manager="transaction"/> </beans>
2.开启注解
service层
package com.hp.service; import com.hp.bean.Demo; import com.hp.dao.DemoDao; import com.hp.util.MyException; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.jdbc.core.JdbcTemplate; import org.springframework.stereotype.Service; import org.springframework.transaction.TransactionStatus; import org.springframework.transaction.annotation.Transactional; import org.springframework.transaction.support.TransactionCallbackWithoutResult; import org.springframework.transaction.support.TransactionTemplate; import javax.annotation.Resource; import java.util.Map; @Service public class DemoService { @Resource DemoDao demoDao; @Transactional//如果是XML方式@Transactional不需要 public void transfer(String in,Double money,String out) { //转出人金额 double num = demoDao.getMoneyById(in); System.out.println(num); //转出人 if(num != 0) { demoDao.update("update demo set money = money - ? where name = ?", money, in); }else { throw new MyException("余额为空"); } // System.out.println(1/0); //收款人 if(num >= money){ demoDao.update("update demo set money = money + ? where name = ?",money,out); }else { throw new MyException("余额不足"); } } public Map login(String name){ Map demo = demoDao.login("select * from demo where name = ?",name); System.out.println(demo); return demo; } }
dao层
package com.hp.dao; import com.hp.bean.Demo; import com.hp.util.MyException; import org.springframework.jdbc.core.JdbcTemplate; import org.springframework.stereotype.Component; import org.springframework.stereotype.Repository; import javax.annotation.Resource; import java.util.Map; @Repository public class DemoDao { @Resource JdbcTemplate jdbcTemplate; public double getMoneyById(String in){ double d = jdbcTemplate.queryForObject("select money from demo where name=?",new Object[] {in},double.class); return d; } public int update(String sql,Object...obj){ int i = jdbcTemplate.update(sql, obj); return i; } public Map login(String sql,String name){ Map demo = jdbcTemplate.queryForMap(sql,name); return demo; } }
bean实体类
package com.hp.bean; import lombok.Data; @Data public class Demo { private Integer id; private String name; private Double money; public Demo(){ } public Demo(Integer id, String name, Double money) { this.id = id; this.name = name; this.money = money; } }
自定义异常类
/** * 自定义异常类 运行时异常 * */ public class MyException extends RuntimeException{ public MyException(String msg) { super(msg); } }
测试
import com.hp.bean.Demo; import com.hp.service.DemoService; import org.junit.Test; import org.junit.runner.RunWith; import org.springframework.test.context.ContextConfiguration; import org.springframework.test.context.junit4.SpringJUnit4ClassRunner; import javax.annotation.Resource; import java.util.Map; import java.util.Scanner; @RunWith(SpringJUnit4ClassRunner.class) @ContextConfiguration(locations = "classpath:applicationContext.xml") public class DemoTest { @Resource DemoService demoService; @Test public void test(){ System.out.println("登录"); demoService.login("张三"); try { demoService.transfer("张三",200.00,"李四"); System.out.println("转账成功!"); } catch (Exception e) { System.out.println(e.getMessage()+",转账失败!"); } } }