事务操作

什么是事务

  • 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接口,代表事务管理器,这个接口针对不同的框架提供了不同的实现类
      Spring事务接口

注解实现声明式事务管理

  • 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:事务传播行为

    • 多事务方法直接进行调用,这个过程中事务是如何进行管理的
      事务的传播行为
      事务的7种传播行为
  • 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; 
		}
	}

  • 0
    点赞
  • 0
    收藏
    觉得还不错? 一键收藏
  • 1
    评论

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论 1
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值