框架学习之Spring的事务管理

一.事务

    1. 事务:指的是逻辑上一组操作,组成这个事务的各个执行单元,要么一起成功,要么一起失败!
    2. 事务的特性
        * 原子性
        * 一致性
        * 隔离性
        * 持久性
    
    3. 如果不考虑隔离性,引发安全性问题
        * 读问题:
            * 脏读:
            * 不可重复读:
            * 虚读:
        
        * 写问题:
            * 丢失更新:
    
    4. 如何解决安全性问题
        * 读问题解决,设置数据库隔离级别
        * 写问题解决可以使用 悲观锁和乐观锁的方式解决

二.Spring框架的JDBC模板技术

    1. Spring框架中提供了很多持久层的模板类来简化编程,使用模板类编写程序会变的简单
    2. 提供了JDBC模板,Spring框架提供的
        * JdbcTemplate类
    
    3. Spring框架可以整合Hibernate框架,也提供了模板类
        * HibernateTemplate类

三.JDBC的模板类应用

1. 步骤一:创建数据库的表结构
        create database spring_day03;
        use spring_day03;
        create table t_account(
            id int primary key auto_increment,
            name varchar(20),
            money double
        );
    
    2. 引入开发的jar包
        * 先引入IOC基本的6个jar包
            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
        * 再引入Spring-aop的jar包
             spring-aop-4.2.4.RELEASE.jar
             com.springsource.org.aopalliance-1.0.0.jar
             com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar
             spring-aspects-4.2.4.RELEASE.jar
            
        * 最后引入JDBC模板需要的jar包
            * MySQL数据库的驱动包
            * spring-jdbc-4.2.4.RELEASE.jar
            * spring-tx-4.2.4.RELEASE.jar
                * 想要做测试再引入一个测试的包
            spring-test-4.2.4.RELEASE.jar

3.创建Spring的配置文件

       * 步骤一:Spring管理内置的连接池
            <bean id="dataSource" class="org.springframework.jdbc.datasource.DriverManagerDataSource">
                <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
                <property name="url" value="jdbc:mysql:///spring_day03"/>
                <property name="username" value="root"/>
                <property name="password" value="root"/>
            </bean>

----------Spring框架管理开源的连接池--------------------------------------------------------------------------------------

1. 管理DBCP连接池
        * 先引入DBCP的2个jar包   
            在我们下载的Spring开发包中的依赖文件夹中可以找到这两个jar包
            资料\spring-framework-3.0.2.RELEASE-dependencies 这个依赖的文件夹中找到DBCP的两个jar包
            * com.springsource.org.apache.commons.dbcp-1.2.2.osgi.jar 
            * com.springsource.org.apache.commons.pool-1.5.3.jar
        
        * 编写配置文件
            <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource">
                <property name="driverClassName" value="com.mysql.jdbc.Driver"/>
                <property name="url" value="jdbc:mysql:///spring_day03"/>
                <property name="username" value="root"/>
                <property name="password" value="root"/>
            </bean>
    
    2. 管理C3P0连接池
        * 先引入C3P0的jar包
            在 资料\spring-framework-3.0.2.RELEASE-dependencies 这个依赖的文件夹中找到C3P0这个jar包
            * com.springsource.com.mchange.v2.c3p0-0.9.1.2.jar
        
        * 编写配置文件
            <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>
    -------------------------------------------------------------------------------------------------------------------------------------
        
        * 步骤二:Spring管理模板类
            <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
                <property name="dataSource" ref="dataSource"/>
            </bean>

4.编写测试程序

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext.xml")
public class Demo2 {
				
    @Resource(name="jdbcTemplate")
	private JdbcTemplate jdbcTemplate;
				
    @Test
    public void run2(){
		jdbcTemplate.update("insert into t_account values (null,?,?)", "测试2",10000);
	}
}

四.Spring框架的JDBC模板的简单操作

1. 增删改查的操作

@RunWith(SpringJUnit4ClassRunner.class)	@ContextConfiguration("classpath:applicationContext.xml")
public class SpringDemo3 {
			
	@Resource(name="jdbcTemplate")
	private JdbcTemplate jdbcTemplate;
			
    @Test
	// 插入操作
	public void demo1(){
		jdbcTemplate.update("insert into account values (null,?,?)", "冠希",10000d);
	}
			
	@Test
	// 修改操作
	public void demo2(){
		jdbcTemplate.update("update account set name=?,money =? where id = ?", "思雨",10000d,5);
	}
			
	@Test
	// 删除操作
	public void demo3(){
		jdbcTemplate.update("delete from account where id = ?", 5);
	}
			
	@Test
	// queryForObject()方法 查询一条记录 也就是返回一行记录  写一个JavaBean 然后查询一条数据,帮你封装进这个JavaBean
	public void demo4(){
	  Account account = jdbcTemplate.queryForObject("select * from account where id = ?", new BeanMapper(), 1);
	  System.out.println(account);
	}
			
	@Test
	// 查询所有记录 使用query方法 直接返回一个集合 这个集合装有所有记录
	public void demo5(){
	    List<Account> list = jdbcTemplate.query("select * from t_account", new BeanMapper());
		for (Account account : list) {
		    System.out.println(account);
		}
	}
}
		

五.Spring框架的事务管理相关的类和API

   1. PlatformTransactionManager接口        -- 平台事务管理器.(真正管理事务的类)。该接口有具体的实现类,根据不同的持久层框架,需要选择不同的实现类!
    2. TransactionDefinition接口            -- 事务定义信息.(事务的隔离级别,传播行为,超时,只读)
    3. TransactionStatus接口                -- 事务的状态
    
    4. 总结:上述对象之间的关系:平台事务管理器真正管理事务对象.根据事务定义的信息TransactionDefinition 进行事务管理,在管理事务中产生一些状态.将状态记录到TransactionStatus中
    
    5. PlatformTransactionManager接口中实现类和常用的方法
        1. 接口的实现类
            * 如果使用的Spring的JDBC模板或者MyBatis框架,需要选择DataSourceTransactionManager实现类
            * 如果使用的是Hibernate的框架,需要选择HibernateTransactionManager实现类
        
        2. 该接口的常用方法
            * void commit(TransactionStatus status) 
            * TransactionStatus getTransaction(TransactionDefinition definition) 
            * void rollback(TransactionStatus status) 
     
    6. TransactionDefinition
        1. 事务隔离级别的常量
            * static int ISOLATION_DEFAULT                     -- 采用数据库的默认隔离级别
            * static int ISOLATION_READ_UNCOMMITTED 
            * static int ISOLATION_READ_COMMITTED 
            * static int ISOLATION_REPEATABLE_READ 
            * static int ISOLATION_SERIALIZABLE 
         
        2. 事务的传播行为常量(不用设置,使用默认值)
            * 先解释什么是事务的传播行为:解决的是业务层之间的方法调用!!
            
            * 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出现异常,运行客户根据需求回滚(选择回滚到保存点或者是最初始状态)

六.搭建事务管理转账案例的环境

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个包
             spring-aop-4.2.4.RELEASE.jar
                 com.springsource.org.aopalliance-1.0.0.jar
             com.springsource.org.aspectj.weaver-1.6.8.RELEASE.jar
             spring-aspects-4.2.4.RELEASE.jar
        * C3P0的1个包
            com.springsource.com.mchange.v2.c3p0-0.9.1.2.jar
        * MySQL的驱动包
            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   日志的配置文件
            
            * 引入applicationContext.xml   注意约束要写全了

                <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>
    
            </beans>
    3. 步骤三:创建对应的包结构和类
        * com.luo.demo1
            * AccountService
            * AccountServlceImpl
            * AccountDao
            * AccountDaoImpl

4. 步骤四:引入Spring的配置文件,将类配置到Spring中
        <bean id="accountService" class=" com.luo.demo1.AccountServiceImpl">
        </bean>
        
        <bean id="accountDao" class=" com.luo.demo1.AccountDaoImpl">
        </bean>

5. 步骤五:在业务层注入DAO ,在DAO中注入JDBC模板(强调:简化开发,以后DAO可以继承JdbcDaoSupport类)

<bean id="accountService" class="com.luo.demo1.AccountServiceImpl">
            <property name="accountDao" ref="accountDao"/>
 </bean>
        
  <bean id="accountDao" class="com.luo.demo1.AccountDaoImpl">
            <property name="dataSource" ref="dataSource"/>
   </bean>

6. 步骤六:编写DAO和Service中的方法

public class AccountDaoImpl extends JdbcDaoSupport implements AccountDao {
	public void outMoney(String out, double money) {
	    this.getJdbcTemplate().update("update t_account set money = money - ? where name = ?", money,out);
	}
	public void inMoney(String in, double money) {
		this.getJdbcTemplate().update("update t_account set money = money + ? where name = ?", money,in);
	}
}

7. 步骤七:编写测试程序.

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext.xml")
public class Demo1 {	
	@Resource(name="accountService")
	private AccountService accountService;
			
	@Test
	public void run1(){
		accountService.pay("张三", "李四", 1000);
	}
}

七.Spring框架的事务管理的分类

1. Spring的编程式事务管理(不推荐使用)
            * 通过手动编写代码的方式完成事务的管理(不推荐)
   2. Spring的声明式事务管理(底层采用AOP的技术)
            * 通过一段配置的方式完成事务的管理(重点掌握注解的方式)

八.事务管理之声明式事务管理(基于AspectJ的XML方式)

    1. 步骤一:恢复转账开发环境 留下C3P0的配置,留下平台事务管理器的配置,留下service dao 这两个类的配置    
    2. 步骤二:引入AOP的开发包  
    3. 步骤三:配置事务管理器(Spring使用PlatformTransactionManager接口来管理事务,所以咱们需要使用到他的实现类)
        <!-- 配置事务管理器 -->

            <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
                <!--因为Spring的事务管理器要管理事务,他得拿到数据库连接对象,因为只要连接对象才能提交或回滚事务,
                而连接对象又在C3PO连接池中,所以我们把连接池注入到Spring的事务管理器中,让他帮我们来管理事务. -->

                <property name="dataSource" ref="dataSource"/>
            </bean>

    4.步骤四:在业务层使用模板管理事务:

 // 注入事务模板对象
private TransactionTemplate transactionTemplate;
public void setTransactionTemplate(TransactionTemplate transactionTemplate) {
   this.transactionTemplate = transactionTemplate;
 }          
public void pay(final String out, final String in, final double money) {
   transactionTemplate.execute(new TransactionCallbackWithoutResult() {               
      protected void doInTransactionWithoutResult(TransactionStatus status) {
          // 扣钱
           accountDao.outMoney(out, money);
           int a = 10/0;
           // 加钱
          accountDao.inMoney(in, money);
      }
     });
 }

5. 步骤五:配置事务增强  注意顺序先配置事务增强,再去配置切面 
        <!-- 配置事务增强 关联上事务管理器-->
        <tx:advice id="txAdvice" transaction-manager="transactionManager">
            <tx:attributes>
                <!--
                    name        :绑定事务的方法名,可以使用通配符,可以配置多个。
                    propagation    :传播行为
                    isolation    :隔离级别
                    read-only    :是否只读 比如查询可以不用开启事务,那么你可以设置为只读
                    timeout        :超时信息
                    rollback-for:发生哪些异常回滚.
                    no-rollback-for:发生哪些异常不回滚.
                 -->
                <!-- 哪些方法加事务 tx:method 可以出现多个,你可以配置多个方法 方法名也可以通配 比如 pay* -->
                <tx:method name="pay" propagation="REQUIRED"/>
            </tx:attributes>
        </tx:advice>

6. 步骤六:配置AOP的切面
        <!-- 配置AOP切面产生代理 -->
        <aop:config>
                <aop:advisor advice-ref="txAdvice" pointcut="execution(* com.luo.demo1.AccountServiceImpl.pay(..))"/>
        </aop:config>

* 注意:如果是自己编写的切面,使用<aop:aspect>标签来配置我们写的切面类,如果是Spring框架提供的切面,配置切面类使用<aop:advisor>标签。

7. 步骤七:编写测试类

  @RunWith(SpringJUnit4ClassRunner.class)
  @ContextConfiguration("classpath:applicationContext.xml")
  public class Demo2 {
 
      @Resource(name="accountService")
      private AccountService accountService;
            
      @Test
      public void run1(){
           accountService.pay("冠希", "美美", 1000);
      }
   }

九.事务管理之声明式事务管理(基于AspectJ的注解方式)

1.步骤1-3同上

2.步骤三:在配置文件中开启注解事务,把平台事务管理器传进去
        <!-- 开启注解事务 -->
        <tx:annotation-driven transaction-manager="transactionManager"/>

3.步骤四:在业务层上添加一个注解:@Transactional
    @Transactional 这个注解 加在类上,那么这个类中所有的方法都有事务了
    如果加在某个方法上,那这个方法就有事务了,一般我们加在类上
    @Transactional 加在方法上这个注解里面里 有一些属性就可以设置,比如 隔离级别,事务的传播行为等等

4. 编写测试类

@RunWith(SpringJUnit4ClassRunner.class)
@ContextConfiguration("classpath:applicationContext3.xml")
public class Demo3 {
			
    @Resource(name="accountService")
	private AccountService accountService;
			
	@Test
	public void run1(){
		accountService.pay("冠希", "美美", 1000);
	}
}

猜你喜欢

转载自blog.csdn.net/Luna_ll/article/details/81543470