今天内容
- Spring框架的事务管理
技术分析之Spring框架的事务管理相关的类和API
1,PlatformTransactionManager接口 -- 平台事务管理器.(真正管理事务的类)。该接口有具体的实现类,根据不同的持久层框架,需要选择不同的实现类!
2,TransactionDefinition接口 -- 事务定义信息.(事务的隔离级别,传播行为,超时,只读)
3,TransactionStatus接口 -- 事务的状态
4,总结:上述对象之间的关系:平台事务管理器真正管理事务对象.根据事务定义的信息TransactionDefinition 进行事务管理,在管理事务中产生一些状态.将状态记录到TransactionStatus中
5,PlatformTransactionManager接口中实现类和常用的方法,
① 接口的实现类
- 如果使用的Spring的JDBC模板或者MyBatis框架,需要选择DataSourceTransactionManager实现类
- 如果使用的是Hibernate的框架,需要选择HibernateTransactionManager实现类
② 该接口的常用方法
- void commit(TransactionStatus status)
- TransactionStatus getTransaction(TransactionDefinition definition)
- void rollback(TransactionStatus status)
6,TransactionDefinition,
① 事务隔离级别的常量
- static int ISOLATION_DEFAULT -- 采用数据库的默认隔离级别
- static int ISOLATION_READ_UNCOMMITTED --
- static int ISOLATION_READ_COMMITTED
- static int ISOLATION_REPEATABLE_READ
- static int ISOLATION_SERIALIZABLE
② 事务的传播行为常量(不用设置,使用默认值)
先解释什么是事务的传播行为:解决的是业务层之间的方法调用!!
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出现异常,运行客户根据需求回滚(选择回滚到保存点或者是最初始状态)
技术分析之搭建事务管理转账案例的环境(强调:简化开发,以后DAO可以继承JdbcDaoSupport类)
1, 步骤一:创建WEB工程,引入需要的jar包
IOC的6个包
com.springsource.org.apache.commons.logging-1.1.1.jar com.springsource.org.apache.log4j-1.2.15.jar spring-beans-4.2.4.RELEASE.jar spring-context-4.2.4.RELEASE.jar spring-core-4.2.4.RELEASE.jar spring-expression-4.2.4.RELEASE.jar
AOP的4个包
com.springsource.org.aopalliance-1.0.0.jar com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar spring-aop-4.2.4.RELEASE.jar spring-aspects-4.2.4.RELEASE.jar
C3P0的1个包
com.springsource.com.mchange.v2.c3p0-0.9.1.2.jar
MySQL的1个驱动包
mysql-connector-java-5.1.7-bin.jar
JDBC目标2个包
spring-jdbc-4.2.4.RELEASE.jar spring-tx-4.2.4.RELEASE.jar
整合JUnit测试包
spring-test-4.2.4.RELEASE.jar
2, 步骤二:引入配置文件
引入配置文件
引入log4j.properties
在src目录下引入
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:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd">
</beans>
- 在
applicationContext.xml
中添加数据库连接
<!-- 配置C3P0连接池 -->
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="com.mysql.jdbc.Driver"/>
<property name="jdbcUrl" value="jdbc:mysql:///spring_day03"/>
<property name="user" value="root"/>
<property name="password" value="root"/>
</bean>
3, 步骤三:创建对应的包结构和类
- com.huachao.demo1
- AccountService
- AccountServlceImpl
- AccountDao
- AccountDaoImpl
package com.huachao.demo1;
public interface AccountService {
public void pay(String out , String in , int money);
}
package com.huachao.demo1;
public class AccountServiceImp implements AccountService {
private AccountDao accountDao;
public void setAccountDao(AccountDao accountDao) {
this.accountDao = accountDao;
}
@Override
public void pay(String out, String in, int money) {
accountDao.outMoney(out, money);
accountDao.inMoney(in, money);
}
}
package com.huachao.demo1;
public interface AccountDao {
/**
* 扣钱
* @param out
* @param money
*/
public void outMoney(String out , int money);
/**
* 加钱
* @param in
* @param money
*/
public void inMoney(String in , int money);
}
package com.huachao.demo1;
import org.springframework.jdbc.core.support.JdbcDaoSupport;
public class AccountDaoImp extends JdbcDaoSupport implements AccountDao {
/**
* 设置JdbcTemplate的代码每次都要写很麻烦
* 如果能写在父类中,那就可以只写一次
* Spring提供了这样的父类JdbcDaoSupport
*/
// private JdbcTemplate jdbcTemplate;
// public void setTemplate(JdbcTemplate jdbcTemplate) {
// this.jdbcTemplate = jdbcTemplate;
// }
@Override
public void outMoney(String out, int money) {
System.out.println(out+"-"+money+"元");
getJdbcTemplate().update("update t_account set money=money-? where name=?", money,out);
}
@Override
public void inMoney(String in, int money) {
System.out.println(in+"+"+money+"元");
getJdbcTemplate().update("update t_account set money=money+? where name=?", money,in);
}
}
4, 步骤四:引入Spring的配置文件,将类配置到Spring中
5, 步骤五:在业务层注入DAO ,在DAO中注入JDBC模板(强调:简化开发,以后DAO可以继承JdbcDaoSupport类)
6, 步骤六:编写DAO和Service中的方法
7,对于JdbcTemplate有两种方式,一种是在applicationContext.xml中配置,一种是在JdbcDaoSupport
中注入dataSource,JdbcDaoSupport会自己创建JdbcTemplate
对应如下两种配置
方式一
<!-- 方式一 -->
<!-- 配置JDBC模板 -->
<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
<property name="dataSource" ref="dataSource"/>
</bean>
<bean id="accountService" class="com.huachao.demo1.AccountServiceImp">
<property name="accountDao" ref="accountDao"/>
</bean>
<bean id="accountDao" class="com.huachao.demo1.AccountDaoImp">
<property name="jdbcTemplate" ref="jdbcTemplate"/>
</bean>
方式二
<!-- 方式二 -->
<!--
不配置JdbcTemplate的bean
不注入jdbcTemplate,而是注入dataSource
因为在JdbcDaoSupport类中有setDataSource方法,会自己创建一个JdbcTemplate
-->
<!-- 配置业务层和持久层 -->
<bean id="accountService" class="com.huachao.demo1.AccountServiceImp">
<property name="accountDao" ref="accountDao"/>
</bean>
<bean id="accountDao" class="com.huachao.demo1.AccountDaoImp">
<property name="dataSource" ref="dataSource"/>
</bean>
7, 步骤七:编写测试程序.
package com.huachao.demo1;
import javax.annotation.Resource;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.springframework.test.context.ContextConfiguration;
import org.springframework.test.context.junit4.SpringJUnit4ClassRunner;
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext.xml")
public class Demo1 {
@Resource(name="accountService")
private AccountService as;
@Test
public void Run1()
{
as.pay("冠希", "美美", 100);
}
}
注: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:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd">
<!-- 配置C3P0连接池 -->
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="com.mysql.jdbc.Driver"/>
<property name="jdbcUrl" value="jdbc:mysql:///spring_day03"/>
<property name="user" value="root"/>
<property name="password" value="root"/>
</bean>
<!-- 方式一 -->
<!-- 配置JDBC模板 -->
<!-- <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
<property name="dataSource" ref="dataSource"/>
</bean>
<bean id="accountService" class="com.huachao.demo1.AccountServiceImp">
<property name="accountDao" ref="accountDao"/>
</bean>
<bean id="accountDao" class="com.huachao.demo1.AccountDaoImp">
<property name="jdbcTemplate" ref="jdbcTemplate"/>
</bean> -->
<!-- 方式二 -->
<!--
不配置JdbcTemplate的bean
不注入jdbcTemplate,而是注入dataSource
因为在JdbcDaoSupport类中有setDataSource方法,会自己创建一个JdbcTemplate
-->
<!-- 配置业务层和持久层 -->
<bean id="accountService" class="com.huachao.demo1.AccountServiceImp">
<property name="accountDao" ref="accountDao"/>
</bean>
<bean id="accountDao" class="com.huachao.demo1.AccountDaoImp">
<property name="dataSource" ref="dataSource"/>
</bean>
</beans>
技术分析之Spring框架的事务管理的分类
1, Spring的事务管理的分类
- Spring的编程式事务管理(不推荐使用)
- 通过手动编写代码的方式完成事务的管理(不推荐)
2, Spring的声明式事务管理(底层采用AOP的技术)
- 通过一段配置的方式完成事务的管理(重点掌握注解的方式)
技术分析之Spring框架的事务管理之编程式的事务管理(了解)
1,说明:Spring为了简化事务管理的代码:提供了模板类 TransactionTemplate,所以手动编程的方式来管理事务,只需要使用该模板类即可!!
2,手动编程方式的具体步骤如下:
① 步骤一:配置一个事务管理器,Spring使用PlatformTransactionManager接口来管理事务,所以咱们需要使用到他的实现类!!
② 步骤二:配置事务管理的模板
<!-- 配置事务管理器 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
</bean>
<!-- 声明手动编码,提供了类模板,使用该类管理事务比较简单,配置事务管理的模板 -->
<bean id="transactionTemplate" class="org.springframework.transaction.support.TransactionTemplate">
<property name="transactionManager" ref="transactionManager"/>
</bean>
③ 步骤三:在需要进行事务管理的类中,注入事务管理的模板.
<bean id="accountService" class="com.huachao.demo1.AccountServiceImp">
<property name="accountDao" ref="accountDao"/>
<property name="transactionTemplate" ref="transactionTemplate"/>
</bean>
④ 步骤四:在业务层使用模板管理事务:
package com.huachao.demo1;
import org.springframework.transaction.TransactionStatus;
import org.springframework.transaction.support.TransactionCallbackWithoutResult;
import org.springframework.transaction.support.TransactionTemplate;
public class AccountServiceImp implements AccountService {
private AccountDao accountDao;
public void setAccountDao(AccountDao accountDao) {
this.accountDao = accountDao;
}
private TransactionTemplate transactionTemplate;
public void setTransactionTemplate(TransactionTemplate transactionTemplate) {
this.transactionTemplate = transactionTemplate;
}
@Override
public void pay(final String out, final String in, final int money) {
transactionTemplate.execute(new TransactionCallbackWithoutResult() {
@Override
protected void doInTransactionWithoutResult(TransactionStatus arg0) {
accountDao.outMoney(out, money);
int a = 100/0;
accountDao.inMoney(in, money);
}
});
}
}
Spring框架的事务管理之声明式事务管理,即通过配置文件来完成事务管理(AOP思想)
- 声明式事务管理又分成两种方式
- 基于AspectJ的XML方式(重点掌握)
- 基于AspectJ的注解方式(重点掌握)
Spring框架的事务管理之基于AspectJ的XML方式(重点掌握)
1,将上面所有的类文件拷贝到包com.huachao.demo2
中
2, AccountServiceImp.java
package com.huachao.demo2;
public class AccountServiceImp implements AccountService {
private AccountDao accountDao;
public void setAccountDao(AccountDao accountDao) {
this.accountDao = accountDao;
}
@Override
public void pay(final String out, final String in, final int money) {
accountDao.outMoney(out, money);
int a = 100/0;
accountDao.inMoney(in, money);
}
}
3,applicationContext2.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:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd">
<!-- 配置C3P0连接池 -->
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="com.mysql.jdbc.Driver"/>
<property name="jdbcUrl" value="jdbc:mysql:///spring_day03"/>
<property name="user" value="root"/>
<property name="password" value="root"/>
</bean>
<!-- 配置事务管理器 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
</bean>
<!-- 配置业务层和持久层 -->
<bean id="accountService" class="com.huachao.demo2.AccountServiceImp">
<property name="accountDao" ref="accountDao"/>
</bean>
<bean id="accountDao" class="com.huachao.demo2.AccountDaoImp">
<property name="dataSource" ref="dataSource"/>
</bean>
</beans>
注:以上步骤将demo2包下的转账操作恢复为不带事务的
4,步骤一:恢复转账开发环境
5,步骤二:引入AOP的开发包,
6,步骤三:配置事务管理器
<!-- 配置事务管理器 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
</bean>
7, 步骤四:配置事务增强
<!-- 配置事务增强 -->
<tx:advice id="txAdvice" transaction-manager="transactionManager">
<tx:attributes>
<!--
name :绑定事务的方法名,可以使用通配符,可以配置多个。
propagation :传播行为
isolation :隔离级别
read-only :是否只读
timeout :超时信息
rollback-for:发生哪些异常回滚.
no-rollback-for:发生哪些异常不回滚.
-->
<!-- 哪些方法加事务 -->
<tx:method name="pay" propagation="REQUIRED"/>
</tx:attributes>
</tx:advice>
7,步骤五:配置AOP的切面
<!-- 配置AOP切面产生代理 -->
<aop:config>
<aop:advisor advice-ref="myAdvice" pointcut="execution(* com.itheima.demo2.AccountServiceImpl.pay(..))"/>
</aop:config>
- 注意:如果是自己编写的切面,使用<aop:aspect>标签,如果是系统制作的,使用<aop:advisor>标签。
8,步骤六:编写测试类
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext2.xml")
public class Demo2 {
@Resource(name="accountService")
private AccountService accountService;
@Test
public void run1(){
accountService.pay("冠希", "美美", 1000);
}
}
注:applicationContext2.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:aop="http://www.springframework.org/schema/aop"
xmlns:tx="http://www.springframework.org/schema/tx"
xsi:schemaLocation="http://www.springframework.org/schema/beans
http://www.springframework.org/schema/beans/spring-beans.xsd
http://www.springframework.org/schema/context
http://www.springframework.org/schema/context/spring-context.xsd
http://www.springframework.org/schema/aop
http://www.springframework.org/schema/aop/spring-aop.xsd
http://www.springframework.org/schema/tx
http://www.springframework.org/schema/tx/spring-tx.xsd">
<!-- 配置C3P0连接池 -->
<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
<property name="driverClass" value="com.mysql.jdbc.Driver"/>
<property name="jdbcUrl" value="jdbc:mysql:///spring_day03"/>
<property name="user" value="root"/>
<property name="password" value="root"/>
</bean>
<!-- 配置事务管理器 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
</bean>
<!-- 声明事务(采用xml配置文件的方式) -->
<!-- 先配置通知 -->
<tx:advice id="myAdvisor" transaction-manager="transactionManager">
<tx:attributes>
<tx:method name="pay" propagation="REQUIRED"/>
</tx:attributes>
</tx:advice>
<!-- 配置AOP切面:如果是自检编写的AOP使用aop:aspect配置
使用的是spring框架提供的通知aop:advisor
-->
<aop:config>
<aop:advisor advice-ref="myAdvisor" pointcut="execution(public * com.huachao.demo2.AccountServiceImp.pay(..))"/>
</aop:config>
<!-- 配置业务层和持久层 -->
<bean id="accountService" class="com.huachao.demo2.AccountServiceImp">
<property name="accountDao" ref="accountDao"/>
</bean>
<bean id="accountDao" class="com.huachao.demo2.AccountDaoImp">
<property name="dataSource" ref="dataSource"/>
</bean>
</beans>
Spring框架的事务管理之基于AspectJ的注解方式(重点掌握,最简单的方式)
0,按上面的方法,拷贝一份代码到com.huachao.demo3
包下,恢复位不带事务的转账操作
1, 步骤一:恢复转账的开发环境
2, 步骤二:配置事务管理器
<!-- 配置事务管理器 -->
<bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
<property name="dataSource" ref="dataSource"/>
</bean>
3, 步骤三:开启注解事务
<!-- 开启注解事务 -->
<tx:annotation-driven transaction-manager="transactionManager"/>
4, 步骤四:在业务层上添加一个注解:@Transactional
5, 编写测试类
@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext3.xml")
public class Demo3 {
@Resource(name="accountService")
private AccountService accountService;
@Test
public void run1(){
accountService.pay("冠希", "美美", 1000);
}
}
Eclipse需要做设置
- 统一工作空间的编码,选择UTF-8
- 把创建JSP页面的编码修改UTF-8
- 重新配置Tomcat服务器
- 先配置Tomcat服务器
- 选择服务器 --> open --> 选择发布项目的目录(webapps目录)
- SSH自己配置约束