什么是事务
- 1、事务是数据库操作的最基本单元,逻辑上在一组操作中,要么全部成功,如果有一个操作失败那么全部操作都失败
- 2、比较经典的场景就列如:银行转账
事务的特性(ACID)
- 1、原子性:要么全部操作成功,有一个操作失败全部操作失败
- 2、一致性:事务发生前后总量不变
- 3、隔离性:不同的人操作同一条事务记录不会相互影响
- 4、持久性:事务结束后最终会被保存
搭建事务操作环境
- 1、创建数据库及表
- 2、创建service 和 搭建dao,完成对象创建和注入
- (1)在service中注入dao,在dao中注入JdbcTemplate,在JdbcTemplate中注入DataSource
//在配置文件中需要开启注解扫描和JdbcTemplate中注入DataSource
...
《不添加事务的转账方式》
//dao
@Repository
public class UserDao{
@Autowired //注入jdbcTemplate
private JdbcTemplate jdbcTemplate;
//创建多钱的方法 (lucy 转账100 给 mary)
public void addMoney(){
String sql = "update user set money = moeny - ? where username = ?";
jdbcTemplate.update(sql,100,"lucy");
}
//创建少钱的方法
public void reduceMoney(){
String sql = "update user set money = moeny + ? where username = ?";
jdbcTemplate.update(sql,100,"mary");
}
}
//service
@Service
public class UserService{
@Autowired
private UserDao userdao;
//转账的方法
public void accountMoney(){
//lucy少100
userdao.addMoney();
//可能发生异常
int a = 10/0;
//mary多100
userdao.reduceMoney();
}
}
《当转账之间发生异常时,可能造成lucy钱少并且mary钱不会多的问题》
- 当发生如上的问题时就需要事务处理该类问题
- 使用事务操作转账过程说明
//在配置文件中需要开启注解扫描和JdbcTemplate中注入DataSource
...
《添加事务的转账方式》
//dao
@Repository
public class UserDao{
@Autowired //注入jdbcTemplate
private JdbcTemplate jdbcTemplate;
//创建多钱的方法 (lucy 转账100 给 mary)
public void addMoney(){
String sql = "update user set money = moeny - ? where username = ?";
jdbcTemplate.update(sql,100,"lucy");
}
//创建少钱的方法
public void reduceMoney(){
String sql = "update user set money = moeny + ? where username = ?";
jdbcTemplate.update(sql,100,"mary");
}
}
//service
@Service
public class UserService{
@Autowired
private UserDao userdao;
//转账的方法
public void accountMoney(){
try{
#第一步:开启事务
#第二步:进行业务操作
//lucy少100
userdao.addMoney();
//可能发生异常
int a = 10/0;
//mary多100
userdao.reduceMoney();
#第三步:没有发生异常,提交事务
}catch(Exception e){
#第三步:发生异常,回滚事务
}
}
}
Spring事务管理的介绍
- 1、一般事务都是添加在service业务处理层上
- 2、在spring进行事务管理操作
- 有两种方式
- 编程式事务管理(一般不使用),使用编程处理异常,代码臃肿
- 声明式事务管理(使用)
- 有两种方式
- 3、声明式事务管理
- 基于注解方式实现(使用方便)
- 基于xml配置文件方式实现
- 4、在Spring进行声明式事务管理,底层使用的是AOP原理
- 5、Spring事务的API
- 提供了PlatformTransactionManager接口,代表事务管理器,这个接口针对不同的框架提供了不同的实现类
- 提供了PlatformTransactionManager接口,代表事务管理器,这个接口针对不同的框架提供了不同的实现类
注解实现声明式事务管理
- 1、spring配置文件中配置事务管理器
- 2、在spring配置文件中开启事务注解
- 引入tx命名空间(如前几篇文章)
- 3、 在service类上(类方法上)添加事务注解
//在配置文件中需要开启注解扫描和JdbcTemplate中注入DataSource
...
《添加事务的转账方式》
//dao
@Repository
public class UserDao{
@Autowired //注入jdbcTemplate
private JdbcTemplate jdbcTemplate;
//创建多钱的方法 (lucy 转账100 给 mary)
public void addMoney(){
String sql = "update user set money = moeny - ? where username = ?";
jdbcTemplate.update(sql,100,"lucy");
}
//创建少钱的方法
public void reduceMoney(){
String sql = "update user set money = moeny + ? where username = ?";
jdbcTemplate.update(sql,100,"mary");
}
}
//service
@Service
@Transactional //事务注解,该注解可以添加到类上,也可添加到相应方法上
public class UserService{
@Autowired
private UserDao userdao;
//转账的方法
public void accountMoney(){
try{
//lucy少100
userdao.addMoney();
//可能发生异常
int a = 10/0;
//mary多100
userdao.reduceMoney();
}catch(Exception e){
}
}
}
@Transactional一些参数配置
-
1、propagation:事务传播行为
- 多事务方法直接进行调用,这个过程中事务是如何进行管理的
- 多事务方法直接进行调用,这个过程中事务是如何进行管理的
-
2、isolation:事务隔离级别
- 事务有特性成为隔离性,多事务操作之间不会产生影响。不考虑隔离性会产生很多问题
- 有三个读问题:脏读、不可重复读、虚(幻)读。
- 脏读:一个未提交事务读取到另一个未提交事务的数据
- 不可重复读:一个未提交事务读取到另一个提交事务修改的数据
- 虚(幻)读:一个未提交事务读取到另一个提交事务修改数据
- 解决:通过设置事务隔离级别,解决读问题
-
3、timeout:超时时间
- 事务需要在设置超时时间内提交事务,否则会进行回滚事务
- 默认值是 -1,设置时间是以秒为单位
-
4、readOnly:是否只读
- 读:查询操作,写:添加、修改、删除操作
- 默认值是false,表示可以查、增删改
- 为true时,只可以查
-
5、rollbackFor:回滚
- 设置出现哪些异常进行事务回滚
-
6、noRollbackFor:不回滚
- 设置出现那些异常不进行事务回滚
完全注解方式
- 创建配置类代替配置文件
//创建配置类
@Configuration
@ComponentScan(basePackages = {"com.xn"}) //开启扫描
@EnableTransactionManagement //开启事务
public class TxConfig{
@Bean
public DruidDataSource getD ruidDataSource(){
DruidDataSource dataSource= new DruidDataSource();
dataSource.setDriverClassName("com.mysql.jdbc.Driver");
dataSource.setUrl("jdbc:mysql:///user");
dataSource.setUsername("root");
dataSource.setPassword("root");
return dataSource;
}
@Bean
public JdbcTemplate getJdbcTemplate(DataSource dataSource){
//在ioc容器中根据类型找到DataSource
JdbcTemplate jdbcTemplate = new JdbcTemplate();
//注入dataSouce
jdbcTemplate.setDataSource(dataSource);
return jdbcTemplate;
}
//创建事务管理器对象
@Bean
public DataSourceTransactionManager getDataSourceTransactionManager(DataSource dataSource){
DataSourceTransactionManager transactionManager = new DataSourceTransactionManager();
transactionManager.setDataSource(dataSource);
return transactionManager;
}
}