
1 事务管理
1.1 事务回顾
事务简介
一组业务ABCD操作,要么全部成功,要么全部不成功。
事务特性:ACID
Ø 原子性:整体 【原子性是指事务包含的所有操作要么全部成功,要么全部失败】
Ø 一致性:数据 【一个事务执行之前和执行之后都必须处于一致性状态】
Ø 隔离性:并发 【对于任意两个并发的事务T1和T2,在事务T1看来,T2要么在T1开始之前就已经结束,要么在T1结束之后才开始,这样每个事务都感觉不到有其他事务在并发地执行。】
Ø 持久性:结果 【持久性是指一个事务一旦被提交了,那么对数据库中的数据的改变就是永久性的】
隔离问题
Ø 脏读:一个事务读到另一个事务未提交的内容【读取未提交内容】
在该隔离级别,所有事务都可以看到其他未提交事务的执行结果。本隔离级别很少用于实际应用,因为它的性能也不比其他级别好多少。
Ø 不可重复读:一个事务读到另一个事务已提交的内容(insert)【读取提交内容】
这是大多数数据库系统的默认隔离级别(但不是MySQL默认的)。它满足了隔离的简单定义:一个事务只能看见已经提交事务所做的改变。
Ø 虚读(幻读):一个事务读到另一个事务已提交的内容(update)
这是MySQL的默认事务隔离级别,它确保同一事务的多个实例在并发读取数据时,会看到同样的数据行。不过理论上,这会导致另一个棘手的问题:幻读 (Phantom Read)。简单的说,幻读指当用户读取某一范围的数据行时,另一个事务又在该范围内插入了新行,当用户再读取该范围的数据行时,会发现有新的“幻影” 行。
Ø Serializable(可串行化)
这是最高的隔离级别,它通过强制事务排序,使之不可能相互冲突,从而解决幻读问题。简言之,它是在每个读的数据行上加上共享锁。在这个级别,可能导致大量的超时现象和锁竞争。
隔离级别–解决问题
Ø read uncommittd,读未提交。存在3个问题。
Ø read committed,读已提交。解决:脏读。存在2个问题。
Ø repeatable read ,可重复读。解决:脏读、不可重复读。存在1个问题。
Ø serializable,串行化。单事务。没有问题。
mysql 事务操作–简单
ABCD 一个事务Connection conn = null;try{ //1 获得连接 conn = …; //2 开启事务 conn.setAutoCommit(false); A B C D //3 提交事务 conn.commit();} catche(){ //4 回滚事务 conn.rollback();}
mysql 事务操作–Savepoint
需求:AB(必须),CD(可选) Connection conn = null;Savepoint savepoint = null; //保存点,记录操作的当前位置,之后可以回滚到指定的位置。(可以回滚一部分)try{ //1 获得连接 conn = …; //2 开启事务 conn.setAutoCommit(false); A B savepoint = conn.setSavepoint(); C D //3 提交事务 conn.commit();} catche(){ if(savepoint != null){ //CD异常 // 回滚到CD之前 conn.rollback(savepoint); // 提交AB conn.commit(); } else{ //AB异常 // 回滚AB conn.rollback(); }}
1.2 Spring事务管理介绍
1.2.1 Spring提供的事务jar包
transaction = tx
1.2.2 Jar中的三个顶级接口
![]() |
|---|
| PlatformTransactionManager**:平台事务管理器,spring要管理事务,必须使用事务管理器,进行事务配置时,必须配置事务管理器** |
| TransactionDefinition:事务详情(事务定义、事务属性),spring用于确定事务具体详情,例如:隔离级别、是否只读、超时时间 等进行事务配置时,必须配置详情。spring将配置项封装到该对象实例。 |
| TransactionStatus:事务状态,spring用于记录当前事务运行状态。例如:是否有保存点,事务是否完成。spring底层根据状态进行相应操作。 |
1.2.3 PlatformTransactionManager 事务管理器
先导入两个包
常用的两个事务管理器
1.2.4 TransactionStatus 事务状态
1.2.5 TransactionDefinition
传播行为:在两个业务之间如何共享事务
| PROPAGATION_REQUIREDrequired , 必须 【默认值】 | 支持当前事务,A如果有事务,B将使用该事务。如果A没有事务,B将创建一个新的事务。 |
|---|---|
| PROPAGATION_SUPPORTSsupports ,支持 | 支持当前事务,A如果有事务,B将使用该事务。如果A没有事务,B将以非事务执行。 |
| PROPAGATION_MANDATORYmandatory ,强制 | 支持当前事务,A如果有事务,B将使用该事务。如果A没有事务,B将抛异常。 |
| PROPAGATION_REQUIRES_NEW requires_new ,必须新的 | 如果A有事务,将A的事务挂起,B创建一个新的事务如果A没有事务,B创建一个新的事务 |
| PROPAGATION_NOT_SUPPORTEDnot_supported ,不支持 | 如果A有事务,将A的事务挂起,B将以非事务执行如果A没有事务,B将以非事务执行 |
| PROPAGATION_NEVERnever,从不 | 如果A有事务,B将抛异常如果A没有事务,B将以非事务执行 |
| PROPAGATION_NESTEDnested ,嵌套 | A和B底层采用保存点机制,形成嵌套事务。 |
掌握:PROPAGATION_REQUIRED、PROPAGATION_REQUIRES_NEW、PROPAGATION_NESTED
1.3 案例:转帐
1.3.1 环境搭建
创建数据库表
create database spring_day3;use spring_day3;create table account( id int primary key auto_increment, username varchar(50), money int);insert into account(username,money) values(‘jack’,’10000’);insert into account(username,money) values(‘rose’,’10000’);
导入jar包
l 核心:4+1
l aop : 4 (aop联盟、spring aop、aspectj规范、spring aspect)
l 数据库:2 (jdbc/tx)
l 驱动:mysql
l 连接池:c3p0
Dao层
Service层
Spring的配置
配置c3p0数据源->dao -> service
测试转帐
1.3.2 手动管理事务【了解】
spring底层使用 TransactionTemplate 事务模板进行操作。
操作
1.service 需要获得 TransactionTemplate
2.spring 配置模板,并注入给service
3.模板需要注入事务管理器
4.配置事务管理器:DataSourceTransactionManager ,需要注入DataSource
了解底层即可,因为以后都是通过aop来配置事务
修改Service
修改spring的配置文件
1.3.3 工厂bean生成代理:半自动
Spring提供 管理事务的代理工厂bean TransactionProxyFactoryBean
修改spring配置文件
transactionAttributes:事务详情prop.key :确定哪些方法使用当前事务配置 prop.text:用于配置事务详情 格式:PROPAGATION,ISOLATION,readOnly,-Exception,+Exception 传播行为 隔离级别 是否只读 异常回滚 异常提交
测试:
1.3.4 基本AOP的事务配置【掌握】
Spring的配置文件
![]() |
|---|
![]() |
测试同上
1.3.5 基本于注解的事务
spring配置
1.3.5 整合Junit
目的:少写一些代码
导入一个spring-test包
2. SSH整合
2.1 web 整合spring
配置tomcat加载spring的配置文件
第一步:需要添加
第二步:在web.xml配置spring的监听
![]() |
|---|
| 出现下面的错误是配置文件加载位置不对,在web.xml改成classpath目录下 |
![]() |
![]() |
第三步:创建Servlet获取Spring的应用上下文件ApplicationContext
2.2 web整合struts+hibernate+spring
整合版本
struts-2.3.33-all
spring-framework-3.0.2.RELEASE
hibernate-distribution-3.6.10.Final-dist
第一步:jar包整合
Struts的jar包
Spring的jar包
基础:4+1 , beans、core、context、expression , commons-logging (struts已经导入)AOP:aop联盟(aopalliance)、spring aop 、aspect规范(aspect.weaver)、spring aspectdb:jdbc、tx测试:testweb开发:spring web驱动:mysql连接池:c3p0整合hibernate:spring orm
Hibernate的jar包
核心包![]() |
|---|
![]() |
required包下的介绍 jpa用于注解开发@Entity @Id |
整合log4j
导入 log4j…jar (struts已经导入)
整合(过渡):slf4j-log4j12-1.7.2.jar
二级缓存
Commons-loggin.jar已经存在
整合包
spring整合hibernate: spring orm
struts 整合spring:struts2-spring-plugin-2.3.15.3.jar
删除重复jar包
第二步:spring整合hibernate的单元测试
创建表
create table t_user( id int primary key auto_increment, username varchar(50), password varchar(32), age int );
po类和映射文件
| public class User { private Integer id; private String username; private String password; private Integer age; |
|---|
dao
Service
hibernate.cfg.xml
applicationContext.xml
单元测试
配置Hibrenate的事务
简化:去除hibernate.cfg.xml文件
第三步:spring整合struts
编写action类,并将其配置给spring ,spring可以注入service
编写struts.xml
表单jsp页面
web.xml 配置
1.确定配置文件contextConfigLocation
2.配置监听器 ContextLoaderListener
3.配置前端控制器 StrutsPrepareAndExecuteFitler
action和spring配置文件
| action中service默认会根据名称注入**默认情况下框架使用的自动装配策略是name,也就是说框架会去 Spring中寻找与action属性名字相同的bean** |
|---|
![]() |
actoin对象由spring创建










jpa用于注解开发@Entity @Id