spring4-5-事務管理


1.簡單介紹

  事務管理是企業級應用程序開發中必不可少的技術,  用來確保數據的完整性和一致性. 事務就是一系列的動作, 它們被當做一個單獨的工作單元. 這些動作要么全部完成, 要么全部不起作用


  事務的四個關鍵屬性(ACID):

  • 原子性(atomicity): 事務是一個原子操作, 由一系列動作組成. 事務的原子性確保動作要么全部完成要么完全不起作用.
  • 一致性(consistency): 一旦所有事務動作完成, 事務就被提交. 數據和資源就處於一種滿足業務規則的一致性狀態中.
  • 隔離性(isolation): 可能有許多事務會同時處理相同的數據, 因此每個事物都應該與其他事務隔離開來, 防止數據損壞.
  • 持久性(durability): 一旦事務完成, 無論發生什么系統錯誤, 它的結果都不應該受到影響. 通常情況下, 事務的結果被寫到持久化存儲器中.

 2.事務管理

  作為企業級應用程序框架, Spring 在不同的事務管理 API 之上定義了一個抽象層. 而應用程序開發人員不必了解底層的事務管理 API, 就可以使用 Spring 的事務管理機制.Spring 既支持編程式事務管理, 也支持聲明式的事務管理.編程式事務管理: 將事務管理代碼嵌入到業務方法中來控制事務的提交和回滾. 在編程式管理事務時, 必須在每個事務操作中包含額外的事務管理代碼. 聲明式事務管理: 大多數情況下比編程式事務管理更好用. 它將事務管理代碼從業務方法中分離出來, 以聲明的方式來實現事務管理. 事務管理作為一種橫切關注點, 可以通過 AOP 方法模塊化. Spring 通過 Spring AOP 框架支持聲明式事務管理.
  Spring 從不同的事務管理 API 中抽象了一整套的事務機制. 開發人員不必了解底層的事務 API, 就可以利用這些事務機制. 有了這些事務機制, 事務管理代碼就能獨立於特定的事務技術了.Spring 的核心事務管理抽象是 interface PlatformTransactionManager   它為事務管理封裝了一組獨立於技術的方法. 無論使用 Spring 的哪種事務管理策略(編程式或聲明式), 事務管理器都是必須的.

3.Spring 中的事務管理器的不同實

org.springframework.jdbc.datasource  Class DataSourceTransactionManager:在應用程序中只需要處理一個數據源, 而且通過 JDBC 存取

org.springframework.jdbc.jta Class  JtaTransactionManager:在 JavaEE 應用服務器上用 JTA(Java Transaction API) 進行事務管理

org.springframework.orm.hibernate3 Class HibernateTransactionManager:用 Hibernate 框架存取數據庫

  事務管理器以普通的 Bean 形式聲明在 Spring IOC 容器中

4.需求:

  

這里將事務添加到bookShopServiceImpl中.

 

5.基於注解方式:

5.1 創建測試代碼

BookShopDao:

 1 package com.l1.spring.jdbc.tx;
 2 
 3 public interface BookShopDao {
 4     //根據書號獲取書的單價
 5     public int findBookPriceByIsbn(String isbn);
 6     //更新書的庫存,庫存減一
 7     public void updateBookStock(String isbn);
 8     
 9     // 更新用戶賬戶余額: balance - price
10     public void updateAccount(String username,int price);
11 }
View Code

 

實現類:

 1 package com.l1.spring.jdbc.tx;
 2 
 3 import org.springframework.beans.factory.annotation.Autowired;
 4 import org.springframework.jdbc.core.JdbcTemplate;
 5 import org.springframework.stereotype.Repository;
 6 
 7 @Repository("bookShopDao")
 8 public class BookShopDaoImpl implements BookShopDao {
 9     
10     @Autowired
11     private JdbcTemplate jdbcTemplate;
12     @Override
13     public int findBookPriceByIsbn(String isbn) {
14         String sql = "select price from book where isbn = ?";
15         return jdbcTemplate.queryForObject(sql, Integer.class, isbn);
16     }
17 
18     @Override
19     public void updateBookStock(String isbn) {
20         //驗證庫存是否足夠
21         String sql1 = "select stock from book_stock where isbn = ? ";
22         int stock = jdbcTemplate.queryForObject(sql1, Integer.class, isbn);
23         if(stock == 0){
24             throw new BookStockException("庫存不足!");
25         }
26         
27         String sql = "update book_stock  set stock = (stock-1) where isbn = ? ";
28         jdbcTemplate.update(sql, isbn);
29     }
30 
31     @Override
32     public void updateAccount(String username, int price) {
33         //驗證賬戶余額是否夠用
34         String sql1  = "select balance from account where username = ?";
35         int balance = jdbcTemplate.queryForObject(sql1, Integer.class, username);
36         if(balance<price){
37             throw new AccountException("賬戶余額不夠!");
38         }
39         
40         String sql = "update account set balance = balance-? where username = ?";
41         jdbcTemplate.update(sql, price,username);
42     }
43 
44 }
View Code

 

BookShopService:

1 package com.l1.spring.jdbc.tx;
2 
3 public interface BookShopService {
4     public void purchase(String username,String isbn);
5 }
View Code

 

實現類:

 1 package com.l1.spring.jdbc.tx;
 2 
 3 import org.springframework.beans.factory.annotation.Autowired;
 4 import org.springframework.beans.factory.annotation.Qualifier;
 5 import org.springframework.stereotype.Service;
 6 import org.springframework.transaction.annotation.Transactional;
 7 
 8 @Service("bookShopService")
 9 public class BookShopServiceImpl implements BookShopService {
10 
11     @Autowired
12     @Qualifier("bookShopDao")
13     private BookShopDao bookShopDao;
14     //添加事務注解
15     @Transactional
16     @Override
17     public void purchase(String username, String isbn) {
18         //1.獲取書的單價
19         int price = bookShopDao.findBookPriceByIsbn(isbn);
20         //2.更新書的庫存
21         bookShopDao.updateBookStock(isbn);
22         //3.更新賬戶余額
23         bookShopDao.updateAccount(username, price);
24     }
25 
26 }
View Code

 

5.2 xml中配置事務管理器:

    <!-- 首先配置事務管理器 -基於注解的方式 -->
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"></property>
    </bean>

 

5.3 xml中引入事務命名空間 tx,啟動事務注解:

    <!-- 引入tx 命名空間,啟用事務注解  -->
    <tx:annotation-driven transaction-manager="transactionManager"/>

 

 

5.4  在需要管理的方法前添加事務注解:

 1 package com.l1.spring.jdbc.tx;
 2 
 3 import org.springframework.beans.factory.annotation.Autowired;
 4 import org.springframework.beans.factory.annotation.Qualifier;
 5 import org.springframework.stereotype.Service;
 6 import org.springframework.transaction.annotation.Transactional;
 7 
 8 @Service("bookShopService")
 9 public class BookShopServiceImpl implements BookShopService {
10 
11     @Autowired
12     @Qualifier("bookShopDao")
13     private BookShopDao bookShopDao;
14     //添加事務注解
15     @Transactional
16     @Override
17     public void purchase(String username, String isbn) {
18         //1.獲取書的單價
19         int price = bookShopDao.findBookPriceByIsbn(isbn);
20         //2.更新書的庫存
21         bookShopDao.updateBookStock(isbn);
22         //3.更新賬戶余額
23         bookShopDao.updateAccount(username, price);
24     }
25 
26 }

 

6.基於xml配置的:

  6.1基本配置:

    <!-- 配置自動掃描組件 -->
    <context:component-scan base-package="com.l1.spring.jdbc.tx"></context:component-scan>
    <!-- 1.導入資源文件 -->
    <context:property-placeholder location="classpath:db.properties"/>
    <!-- 2.配置C3P0數據源 -->
    <bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
        <property name="user" value="${jdbc.user}"></property>
        <property name="password" value="${jdbc.password}"></property>
        <property name="driverClass" value="${jdbc.driverClass}"></property>
        <property name="jdbcUrl" value="${jdbc.jdbcUrl}"></property>
        <property name="initialPoolSize" value="${jdbc.initialPoolSize}"></property>
        <property name="maxPoolSize" value="${jdbc.maxPoolSize}"></property>
    </bean>

    <!-- 3.配置 jdbc template -->
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
        <property name="dataSource" ref="dataSource"></property>
    </bean>
View Code

 

  6.2 配置bean:

    <!-- 1. 配置Bean -->
    <bean id="bookShopDao" class="com.l1.spring.jdbc.tx.xml.BookShopDaoImpl">
        <property name="jdbcTemplate" ref="jdbcTemplate"></property>
    </bean>
    <bean id="bookShopService" class="com.l1.spring.jdbc.tx.xml.BookShopServiceImpl">
        <property name="bookShopDao" ref="bookShopDao"></property>
    </bean>
    <bean id="cashier" class="com.l1.spring.jdbc.tx.xml.CashierImpl">
        <property name="bookShopService"  ref="bookShopService"></property>
    </bean>

 

   6.3 配置事務管理器:

    <!-- 2.配置事務管理器 -->
    <bean id="transactionManager" class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
        <property name="dataSource" ref="dataSource"></property>
    </bean>

 

  6.4 配置事務通知,屬性:

    <!-- 3.配置事務屬性 -->
    <tx:advice id="txAdvice" transaction-manager="transactionManager">
        <tx:attributes>
            <tx:method name="*"/>
        </tx:attributes>
    </tx:advice>

 

  指定事務屬性:

    <!-- 3.配置事務屬性 -->
    <tx:advice id="txAdvice" transaction-manager="transactionManager">
        <tx:attributes>
            <!-- 根據方法名指定事務屬性 -->
            <tx:method name="purchase" propagation="REQUIRES_NEW"/>
            <tx:method name="get*" read-only="true"/>
            <tx:method name="find*" read-only="true"/>
            <tx:method name="*"/>
        </tx:attributes>
    </tx:advice>

 

  6.5 配置事務切入點,將切入點和事務通知關聯起來:

    <!-- 4.配置事務切入點,以及關聯切入點和事務屬性 -->
    <aop:config>
        <aop:pointcut expression="execution(public * com.l1.spring.jdbc.tx.xml.BookShopService.*(..))"
             id="txPoinCut"/>
             
        <aop:advisor advice-ref="txAdvice" pointcut-ref="txPoinCut"/>
    </aop:config>

 

 

 

 

 



  


免責聲明!

本站轉載的文章為個人學習借鑒使用,本站對版權不負任何法律責任。如果侵犯了您的隱私權益,請聯系本站郵箱yoyou2525@163.com刪除。



 
粵ICP備18138465號   © 2018-2025 CODEPRJ.COM