Spring 事务详细讲解
Spring 事务
1. 回忆之前事务知识点
一:事务的概念
将一系列的数据操作捆绑在一起,成为一个整体进行统一管理!
一条或者多条sql语句的集合!
二:事务的ACID特性
原子性(Atomicity):在事务中的操作,要么都执行,要么都不执行!
一致性(Consistency):事务必须保证数据从一个一致性的状态到达另一个一致性的状态!
持久性(Durability):事务一旦提交,对数据库的数据影响是永久性的!
隔离性(Isolation):每个事务之前互不干扰!
三: 事务创建的原则:
1.事务尽可能简短
因为启动事务之后,数据库管理系统需要保留大量资源来保证事务的ACID!
如果是多用户的系统,那么会严重影响我们的系统性能!
2.事务中访问的数据量要小
在并发的情况下,执行事务,事务的访问量越小,各个线程之前对数据的
争夺就越小!
3.查询时尽量不使用事务
因为查询不会修改数据!
4.在事务处理过程中,尽量不要出现用户等待的情况
如果登台事件过程,占用资源太久,有可能造成系统阻塞!
四: 必须掌握事务中的两个属性:
1.Isolation:五种事务隔离级别
01.default:采用数据库默认的事务隔离级别
mysql:repeatable-read(可重复读取)
oracle:read-committed(读提交)
02.repeatable-read(可重复读取):解决了脏读和不可重复读,但是有可能发生幻读(通过其他的锁机制解决)!
03.read-committed(读提交):解决了脏读,但是没解决不可重复读
04.read-uncommitted(读未提交):什么都没有解决!
05.serializable(串行化):解决并发的问题!
2.Propagation:七种事务传播行为
01.required:是spring默认的事务传播行为!
指定的方法必须在事务内执行,
如果没有事务,自动创建事务并执行!
02.supports:有没有事务都执行!
有事务就在事务中执行,否则直接执行!
03.mandatory:如果当前存在事务,就执行该事务,
如果当前不存在事务,就抛出异常。
04.requires_new:总是创建新事务,
如果当前方法存在事务,则当前事务挂起,
直到新创建的事务执行完毕!
05.not_supported:不能在事务中执行,
如果当前方法存在事务,则当前事务挂起!
06.never: 不能在事务中执行,
如果当前方法存在事务,则抛出异常。
07.nested:如果当前存在事务,则在嵌套事务内执行。
如果当前没有事务,则执行与requires_new类似的操作。
requires_new 启动一个新的, 不依赖于环境的 "内部" 事务.
这个事务将被完全 commited 或 rolled back 而不依赖于外部事务,
它拥有自己的隔离范围, 自己的锁, 等等. 当内部事务开始执行时,
外部事务将被挂起,内务事务结束时, 外部事务将继续执行.
另一方面, nested 开始一个 "嵌套的" 事务,
它是已经存在事务的一个真正的子事务. 潜套事务开始执行时,
它将取得一个 savepoint. 如果这个嵌套事务失败,
我们将回滚到此 savepoint. 潜套事务是外部事务的一部分,
只有外部事务结束后它才会被提交.
由此可见, requires_new 和 nested 的最大区别在于,
requires_new 完全是一个新的事务,
而 nested 则是外部事务的子事务,
如果外部事务 commit, 潜套事务也会被 commit,
这个规则同样适用于 roll back.
2. 创建需要的数据库语句
DROP TABLE IF EXISTS `account`;
CREATE TABLE `account` (
`aid` int(10) NOT NULL AUTO_INCREMENT,
`balance` double DEFAULT NULL,
`aname` varchar(20) DEFAULT NULL,
PRIMARY KEY (`aid`)
) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8;
insert into `account`(`aid`,`balance`,`aname`) values (1,10000,'张三');
DROP TABLE IF EXISTS `stock`;
CREATE TABLE `stock` (
`sid` int(10) NOT NULL AUTO_INCREMENT,
`sname` varchar(20) DEFAULT NULL,
`amount` int(10) DEFAULT NULL,
PRIMARY KEY (`sid`)
) ENGINE=InnoDB AUTO_INCREMENT=2 DEFAULT CHARSET=utf8;
insert into `stock`(`sid`,`sname`,`amount`) values (1,'张三',0);
3. 创建dao层和impl
4. 创建service层和impl
5. 创建需要的异常类 和 jdbc.properties文件
6. 使用AspectJ实现事务的管理xml配置文件
<!-- 01.引入 jdbc文件--> <context:property-placeholder location="classpath:jdbc.properties"/> <!-- 02.配置数据源--> <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource"> <property name="driverClassName" value="${jdbc.driverClass}"/> <property name="url" value="${jdbc.jdbcUrl}"/> <property name="username" value="${jdbc.userName}"/> <property name="password" value="${jdbc.password}"/> </bean> <!--03.配置dao层--> <bean id="accountDao" class="com.xdf.dao.impl.AccountDaoImpl"> <property name="dataSource" ref="dataSource"/> </bean> <bean id="stockDao" class="com.xdf.dao.impl.StockDaoImpl"> <property name="dataSource" ref="dataSource"/> </bean> <!--04.配置service层--> <bean id="buyStockService" class="com.xdf.service.BuyStockServiceImpl"> <property name="accountDao" ref="accountDao"/> <property name="stockDao" ref="stockDao"/> </bean> <!--05.无论使用什么方式 都必须创建事务管理器--> <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager"> <property name="dataSource" ref="dataSource"/> </bean> <!--06. 使用AspectJ 实现spring对事务的管理--> <tx:advice id="myAdvice" transaction-manager="transactionManager"> <tx:attributes> <tx:method name="buyStock" isolation="DEFAULT" propagation="REQUIRED" rollback-for="com.xdf.service.BuyStockException"/> </tx:attributes> </tx:advice> <!-- 07.配置切入点--> <aop:config> <aop:pointcut id="myPoint" expression="execution(* *..service.*.*(..))"/> <aop:advisor advice-ref="myAdvice" pointcut-ref="myPoint"/> </aop:config>
测试方法代码
7. 使用事务工厂实现事务的管理xml配置文件
<!--06.使用事务工厂--> <bean id="proxyFactoryBean" class="org.springframework.transaction.interceptor.TransactionProxyFactoryBean"> <!--配置事务管理器--> <property name="transactionManager" ref="transactionManager"/> <!--配置目标对象--> <property name="target" ref="buyStockService"/> <!--配置切点--> <property name="transactionAttributes"> <props> <!--指定切入点 01.事务隔离级别 02.事务传播行为 03.发生什么异常时?是回滚还是提交 - 回滚 +提交 --> <prop key="buyStock">ISOLATION_DEFAULT,PROPAGATION_REQUIRED,-BuyStockException</prop> </props> </property>
测试方法代码
8. 使用注解实现事务的管理xml配置文件
测试代码和AspectJ一样
- php dirname(__FILE__) 获取当前文件的绝对路径
- 洛谷P2939 [USACO09FEB]改造路Revamping Trails(最短路)
- 微信公众平台开发接口PHP SDK完整版
- 我的HTML总结之HTML发展史
- BZOJ 4152: [AMPPZ2014]The Captain(最短路)
- js获取div编辑框,textarea,input text的光标位置,兼容FF和IE
- BZOJ 4289: PA2012 Tax(最短路)
- php QR Code二维码生成类
- BZOJ 3714: [PA2014]Kuglarz(最小生成树)
- 我的HTML总结之表单
- php 二维码生成类
- HDU 2516 取石子游戏(斐波那契博弈)
- angularjs MVC、模块化、依赖注入详解
- BZOJ 2940: [Poi2000]条纹(Multi-Nim)
- java教程
- Java快速入门
- Java 开发环境配置
- Java基本语法
- Java 对象和类
- Java 基本数据类型
- Java 变量类型
- Java 修饰符
- Java 运算符
- Java 循环结构
- Java 分支结构
- Java Number类
- Java Character类
- Java String类
- Java StringBuffer和StringBuilder类
- Java 数组
- Java 日期时间
- Java 正则表达式
- Java 方法
- Java 流(Stream)、文件(File)和IO
- Java 异常处理
- Java 继承
- Java 重写(Override)与重载(Overload)
- Java 多态
- Java 抽象类
- Java 封装
- Java 接口
- Java 包(package)
- Java 数据结构
- Java 集合框架
- Java 泛型
- Java 序列化
- Java 网络编程
- Java 发送邮件
- Java 多线程编程
- Java Applet基础
- Java 文档注释
- vue动态组件的用法
- elementUI中checekBox实现全选和反选以及实现在input中输入空格出现label分割
- centos 安装git
- centos 挂载硬盘
- centos 安装postgrep
- springboot集成PowerJob-openAPI和回调完整流程
- 使用 Redisson 实现的 redis 分布式锁在 SpringBoot 中的简单使用
- SpringBoot 的过滤器 Filter 配置
- CentOS 7 Redis 5.0.8 哨兵模式配置
- SpringBoot 项目的 Maven 多环境打包配置
- webRTC: 基于coturn项目的stun/turn服务器搭建
- 百度地图电子围栏功能
- opencv使用中出现的问题
- 人生苦短,我用k8s--------------Pod资源控制进阶管理
- 人生苦短,我用k8s--------------kuberctl命令合集