Spring第八篇【XML、註解實現事務控制】

前言

本博文主要講解Spring的事務控制,如何使用Spring來對程序進行事務控制….css

通常地,咱們事務控制都是在service層作的。。爲何是在service層而不是在dao層呢??有沒有這樣的疑問…java

service層是業務邏輯層,service的方法一旦執行成功,那麼說明該功能沒有出錯mysql

一個service方法可能要調用dao層的多個方法…若是在dao層作事務控制的話,一個dao方法出錯了,僅僅把事務回滾到當前dao的功能,這樣是不合適的。若是沒有出錯,調用完dao方法就commit了事務,這也是不合適的。spring


事務控制概述

事務控制分爲兩種:sql

  • 編程式事務控制
  • 聲明式事務控制

編程式事務控制

本身手動控制事務,就叫作編程式事務控制。數據庫

  • Jdbc代碼:
    • Conn.setAutoCommite(false); // 設置手動控制事務
  • Hibernate代碼:
    • Session.beginTransaction(); // 開啓一個事務
  • 【細粒度的事務控制: 能夠對指定的方法、指定的方法的某幾行添加事務控制】
  • (比較靈活,但開發起來比較繁瑣: 每次都要開啓、提交、回滾.)

聲明式事務控制

Spring提供對事務的控制管理就叫作聲明式事務控制express

Spring提供了對事務控制的實現。編程

  • 若是用戶想要使用Spring的事務控制,只須要配置就好了
  • 當不用Spring事務的時候,直接移除就好了。
  • Spring的事務控制是基於AOP實現的。所以它的耦合度是很是低的。
  • 【粗粒度的事務控制: 只能給整個方法應用事務,不能夠對方法的某幾行應用事務。
    • (由於aop攔截的是方法。)

Spring給咱們提供了事務的管理器類,事務管理器類又分爲兩種,由於JDBC的事務和Hibernate的事務是不同的markdown

  • Spring聲明式事務管理器類:
    • Jdbc技術:DataSourceTransactionManager
    • Hibernate技術:HibernateTransactionManager

聲明式事務控制

咱們基於Spring的JDBC來作例子吧ide

引入相關jar包

  • AOP相關的jar包【由於Spring的聲明式事務控制是基於AOP的,那麼就須要引入AOP的jar包。】
  • 引入tx名稱空間
  • 引入AOP名稱空間
  • 引入jdbcjar包【jdbc.jar包和tx.jar包】

搭建配置環境

  • 編寫一個接口
public interface IUser {
    void save();
}
  • UserDao實現類,使用JdbcTemplate對數據庫進行操做!
@Repository
public class UserDao implements IUser {

    //使用Spring的自動裝配
    @Autowired
    private JdbcTemplate template;

    @Override
    public void save() {
        String sql = "insert into user(name,password) values('zhong','222')";
        template.update(sql);
    }

}
  • userService
@Service
public class UserService {

    @Autowired
    private UserDao userDao;
    public void save() {

        userDao.save();
    }
}
  • bean.xml配置:配置數據庫鏈接池、jdbcTemplate對象、掃描註解
<?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:c="http://www.springframework.org/schema/c" 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">


    <!--數據鏈接池配置-->
    <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
        <property name="driverClass" value="com.mysql.jdbc.Driver"></property>
        <property name="jdbcUrl" value="jdbc:mysql:///zhongfucheng"></property>
        <property name="user" value="root"></property>
        <property name="password" value="root"></property>
        <property name="initialPoolSize" value="3"></property>
        <property name="maxPoolSize" value="10"></property>
        <property name="maxStatements" value="100"></property>
        <property name="acquireIncrement" value="2"></property>
    </bean>

    <!--掃描註解-->
    <context:component-scan base-package="bb"/>

    <!-- 2. 建立JdbcTemplate對象 -->
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <property name="dataSource" ref="dataSource"></property>
    </bean>

</beans>

前面搭建環境的的時候,是沒有任何的事務控制的。

也就是說,當我在service中調用兩次userDao.save(),即時在中途中有異常拋出,仍是能夠在數據庫插入一條記錄的

  • Service代碼:
@Service
public class UserService {

    @Autowired
    private UserDao userDao;
    public void save() {

        userDao.save();

        int i = 1 / 0;
        userDao.save();
    }
}
  • 測試代碼:
public class Test2 {

    @Test
    public void test33() {
        ApplicationContext ac = new ClassPathXmlApplicationContext("bb/bean.xml");

        UserService userService = (UserService) ac.getBean("userService");
        userService.save();
    }
}

這裏寫圖片描述


XML方式實現聲明式事務控制

首先,咱們要配置事務的管理器類:由於JDBC和Hibernate的事務控制是不一樣的。

<!--1.配置事務的管理器類:JDBC-->
    <bean id="txManage" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">

        <!--引用數據庫鏈接池-->
        <property name="dataSource" ref="dataSource"/>
    </bean>

再而,配置事務管理器類如何管理事務

<!--2.配置如何管理事務-->
    <tx:advice id="txAdvice" transaction-manager="txManage">

        <!--配置事務的屬性-->
        <tx:attributes>
            <!--全部的方法,並非只讀-->
            <tx:method name="*" read-only="false"/>
        </tx:attributes>
    </tx:advice>

最後,配置攔截哪些方法,

<!--3.配置攔截哪些方法+事務的屬性-->
    <aop:config>
        <aop:pointcut id="pt" expression="execution(* bb.UserService.*(..) )"/>
        <aop:advisor advice-ref="txAdvice" pointcut-ref="pt"></aop:advisor>
    </aop:config>

配置完成以後,service中的方法都應該被Spring的聲明式事務控制了。所以咱們再次測試一下:

@Test
    public void test33() {
        ApplicationContext ac = new ClassPathXmlApplicationContext("bb/bean.xml");

        UserService userService = (UserService) ac.getBean("userService");
        userService.save();
    }

這裏寫圖片描述


使用註解的方法實現事務控制

固然了,有的人可能以爲到XML文件上配置太多東西了。Spring也提供了使用註解的方式來實現對事務控制

第一步和XML的是同樣的,必須配置事務管理器類:

<!--1.配置事務的管理器類:JDBC-->
    <bean id="txManage" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">

        <!--引用數據庫鏈接池-->
        <property name="dataSource" ref="dataSource"/>
    </bean>

第二步:開啓以註解的方式來實現事務控制

<!--開啓以註解的方式實現事務控制-->
    <tx:annotation-driven transaction-manager="txManage"/>

最後,想要控制哪一個方法事務,在其前面添加@Transactional這個註解就好了!若是想要控制整個類的事務,那麼在類上面添加就好了。

@Transactional
    public void save() {

        userDao.save();

        int i = 1 / 0;
        userDao.save();
    }

這裏寫圖片描述


事務屬性

其實咱們在XML配置管理器類如何管理事務,就是在指定事務的屬性!咱們來看一下事務的屬性有什麼:

這裏寫圖片描述

對於事務的隔離級別,不清楚的朋友可參考我以前的博文:http://blog.csdn.net/hon_3y/article/details/53760782

事務傳播行爲:

看了上面的事務屬性,沒有接觸過的其實就這麼一個:propagation = Propagation.REQUIRED事務的傳播行爲。

事務傳播行爲的屬性有如下這麼多個,經常使用的就只有兩個:

  • Propagation.REQUIRED【若是當前方法已經有事務了,加入當前方法事務
  • Propagation.REQUIRED_NEW【若是當前方法有事務了,當前方法事務會掛起。始終開啓一個新的事務,直到新的事務執行完、當前方法的事務纔開始】

這裏寫圖片描述

當事務傳播行爲是Propagation.REQUIRED

  • 如今有一個日誌類,它的事務傳播行爲是Propagation.REQUIRED
Class Log{ Propagation.REQUIRED insertLog(); }
  • 如今,我要在保存以前記錄日誌
Propagation.REQUIRED
    Void  saveDept(){
        insertLog();   
        saveDept();
    }

saveDept()自己就存在着一個事務,當調用insertLog()的時候,insertLog()的事務會加入到saveDept()事務中

也就是說,saveDept()方法內始終是一個事務,若是在途中出現了異常,那麼insertLog()的數據是會被回滾的【由於在同一事務內】

Void  saveDept(){
        insertLog();    // 加入當前事務
        .. 異常, 會回滾
        saveDept();
    }

當事務傳播行爲是Propagation.REQUIRED_NEW

  • 如今有一個日誌類,它的事務傳播行爲是Propagation.REQUIRED_NEW
Class Log{ Propagation.REQUIRED insertLog(); }
  • 如今,我要在保存以前記錄日誌
Propagation.REQUIRED
    Void  saveDept(){
        insertLog();   
        saveDept();
    }

當執行到saveDept()中的insertLog()方法時,insertLog()方法發現 saveDept()已經存在事務了,insertLog()會獨自新開一個事務,直到事務關閉以後,再執行下面的方法

若是在中途中拋出了異常,insertLog()是不會回滾的,由於它的事務是本身的,已經提交了

Void  saveDept(){
        insertLog();    // 始終開啓事務
        .. 異常, 日誌不會回滾
        saveDept();
    }
相關文章
相關標籤/搜索