Spring程式設計式和宣告式事務例項講解

SnailClimb發表於2019-03-04

Java面試通關手冊(Java學習指南):github.com/Snailclimb/…

歷史回顧: 可能是最漂亮的Spring事務管理詳解

Spring事務管理

Spring支援兩種方式的事務管理:

  • 程式設計式事務管理: 通過Transaction Template手動管理事務,實際應用中很少使用,
  • 使用XML配置宣告式事務: 推薦使用(程式碼侵入性最小),實際是通過AOP實現

實現宣告式事務的四種方式:

  1. 基於 TransactionInterceptor 的宣告式事務: Spring 宣告式事務的基礎,通常也不建議使用這種方式,但是與前面一樣,瞭解這種方式對理解 Spring 宣告式事務有很大作用。

  2. 基於 TransactionProxyFactoryBean 的宣告式事務: 第一種方式的改進版本,簡化的配置檔案的書寫,這是 Spring 早期推薦的宣告式事務管理方式,但是在 Spring 2.0 中已經不推薦了。

  3. 基於< tx> 和< aop>名稱空間的宣告式事務管理: 目前推薦的方式,其最大特點是與 Spring AOP 結合緊密,可以充分利用切點表示式的強大支援,使得管理事務更加靈活。

  4. 基於 @Transactional 的全註解方式: 將宣告式事務管理簡化到了極致。開發人員只需在配置檔案中加上一行啟用相關後處理 Bean 的配置,然後在需要實施事務管理的方法或者類上使用 @Transactional 指定事務規則即可實現事務管理,而且功能也不必其他方式遜色。

我們今天要將的是使用程式設計式以及基於AspectJ的宣告式和基於註解的事務方式,實現爛大街的轉賬業務。

再來說一下這個案例的思想吧,我們在兩次轉賬之間新增一個錯誤語句(對應銀行斷電等意外情況),如果這個時候兩次轉賬不能成功,則說明事務配置正確,否則,事務配置不正確。

你需要完成的任務:

  • 使用程式設計式事務管理完成轉賬業務
  • 使用基於AspectJ的宣告式事務管理完成轉賬業務
  • 使用基於 @Transactional 的全註解方式事務管理完成轉賬業務

備註:

下面的程式碼是在很久之前,我剛學Sping還沒有接觸Maven的時候寫的,所以我使用的原始新增jar的方式,使用Maven的小夥伴可以自行新增Maven依賴,沒有使用Maven的小夥伴直接使用我下面提供的jar包即可。

jar包地址:連結:pan.baidu.com/s/1tqy-mVKx… 密碼:nid0

專案結構:

專案結構

開發工具:

Myeclipse2017

SQL:


create table `account` (
	`username` varchar (99),
	`salary` int (11)
); 
insert into `account` (`username`, `salary`) values('小王','3000');
insert into `account` (`username`, `salary`) values('小馬','3000');

複製程式碼

(1)程式設計式事務管理

注意: 通過新增/刪除accountMoney() 方法中int i = 10 / 0這個語句便可驗證事務管理是否配置正確。

OrdersDao.java(Dao層)

package cn.itcast.dao;

import org.springframework.jdbc.core.JdbcTemplate;

public class OrdersDao {
	// 注入jdbcTemplate模板物件
	private JdbcTemplate jdbcTemplate;

	public void setJdbcTemplate(JdbcTemplate jdbcTemplate) {
		this.jdbcTemplate = jdbcTemplate;
	}

	// 對資料操作的方法不包含業務操作
	/**
	 * 小王少錢的方法
	 */
	public void reduceMoney() {
		String sql = "update account set salary=salary-? where username=?";
		jdbcTemplate.update(sql, 1000, "小王");
	}

	/**
	 * 小馬多錢的方法
	 */
	public void addMoney() {
		String sql = "update account set salary=salary+? where username=?";
		jdbcTemplate.update(sql, 1000, "小馬");
	}
}
複製程式碼

OrdersService.java(業務邏輯層)

package cn.itcast.service;

import org.springframework.transaction.TransactionStatus;
import org.springframework.transaction.support.TransactionCallback;
import org.springframework.transaction.support.TransactionTemplate;

import cn.itcast.dao.OrdersDao;

public class OrdersService {
	// 注入Dao層物件
	private OrdersDao ordersDao;

	public void setOrdersDao(OrdersDao ordersDao) {
		this.ordersDao = ordersDao;
	}

	// 注入TransactionTemplate物件
	private TransactionTemplate transactionTemplate;

	public void setTransactionTemplate(TransactionTemplate transactionTemplate) {
		this.transactionTemplate = transactionTemplate;
	}

	// 呼叫dao的方法
	// 業務邏輯,寫轉賬業務
	public void accountMoney() {
		transactionTemplate.execute(new TransactionCallback<Object>() {

			@Override
			public Object doInTransaction(TransactionStatus status) {
				Object result = null;
				try {
					// 小馬多1000
					ordersDao.addMoney();
					// 加入出現異常如下面int
					// i=10/0(銀行中可能為突然停電等。。。);結果:小馬賬戶多了1000而小王賬戶沒有少錢
					// 解決辦法是出現異常後進行事務回滾
					int i = 10 / 0;// 事務管理配置後異常已經解決
					// 小王 少1000
					ordersDao.reduceMoney();
				} catch (Exception e) {
					status.setRollbackOnly();
					result = false;
					System.out.println("Transfer Error!");
				}

				return result;
			}
		});

	}
}
複製程式碼

TestService.java(測試方法)

package cn.itcast.service;

import org.junit.Test;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;

public class TestService {
	@Test
	public void testAdd() {
		ApplicationContext context = new ClassPathXmlApplicationContext(
				"beans.xml");
		OrdersService userService = (OrdersService) context
				.getBean("ordersService");
		userService.accountMoney();
	}
}

複製程式碼

配置檔案:

<?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-2.5.xsd  
http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-2.5.xsd  
http://www.springframework.org/schema/aop http://www.springframework.org/schema/aop/spring-aop-2.5.xsd  
http://www.springframework.org/schema/tx http://www.springframework.org/schema/tx/spring-tx-2.5.xsd">
	<!-- 配置c3po連線池 -->
	<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
		<!-- 注入屬性值 -->
		<property name="driverClass" value="com.mysql.jdbc.Driver"></property>
		<property name="jdbcUrl" value="jdbc:mysql://localhost:3306/wangyiyun"></property>
		<property name="user" value="root"></property>
		<property name="password" value="153963"></property>
	</bean>
	<!-- 程式設計式事務管理 -->
	<!-- 配置事務管理器 -->
	<bean id="dataSourceTransactionManager"
		class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
		<!-- 注入dataSource -->
		<property name="dataSource" ref="dataSource"></property>
	</bean>

	<!-- 配置事務管理器模板 -->
	<bean id="transactionTemplate"
		class="org.springframework.transaction.support.TransactionTemplate">
		<!-- 注入真正進行事務管理的事務管理器,name必須為 transactionManager否則無法注入 -->
		<property name="transactionManager" ref="dataSourceTransactionManager"></property>
	</bean>

	<!-- 物件生成及屬性注入 -->
	<bean id="ordersService" class="cn.itcast.service.OrdersService">
		<property name="ordersDao" ref="ordersDao"></property>
		<!-- 注入事務管理的模板 -->
		<property name="transactionTemplate" ref="transactionTemplate"></property>
	</bean>

	<bean id="ordersDao" class="cn.itcast.dao.OrdersDao">
		<property name="jdbcTemplate" ref="jdbcTemplate"></property>
	</bean>
	<!-- JDBC模板物件 -->
	<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
		<property name="dataSource" ref="dataSource"></property>
	</bean>
</beans> 
複製程式碼

(2)基於AspectJ的宣告式事務管理

OrdersService.java(業務邏輯層)

package cn.itcast.service;

import cn.itcast.dao.OrdersDao;

public class OrdersService {
	private OrdersDao ordersDao;

	public void setOrdersDao(OrdersDao ordersDao) {
		this.ordersDao = ordersDao;
	}

	// 呼叫dao的方法
	// 業務邏輯,寫轉賬業務
	public void accountMoney() {
		// 小馬多1000
		ordersDao.addMoney();
		// 加入出現異常如下面int i=10/0(銀行中可能為突然停電等。。。);結果:小馬賬戶多了1000而小王賬戶沒有少錢
		// 解決辦法是出現異常後進行事務回滾
		int i = 10 / 0;// 事務管理配置後異常已經解決
		// 小王 少1000
		ordersDao.reduceMoney();
	}
}

複製程式碼

配置檔案:

	<!-- 配置c3po連線池 -->
	<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
		<!-- 注入屬性值 -->
		<property name="driverClass" value="com.mysql.jdbc.Driver"></property>
		<property name="jdbcUrl" value="jdbc:mysql://localhost:3306/wangyiyun"></property>
		<property name="user" value="root"></property>
		<property name="password" value="153963"></property>
	</bean>
	<!-- 第一步:配置事務管理器 -->
	<bean id="dataSourceTransactionManager"
		class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
		<!-- 注入dataSource -->
		<property name="dataSource" ref="dataSource"></property>
	</bean>

	<!-- 第二步:配置事務增強 -->
	<tx:advice id="txadvice" transaction-manager="dataSourceTransactionManager">
		<!-- 做事務操作 -->
		<tx:attributes>
			<!-- 設定進行事務操作的方法匹配規則 -->
			<!-- account開頭的所有方法 -->
	        <!--
	          propagation:事務傳播行為; 
	          isolation:事務隔離級別;
	          read-only:是否只讀;
	          rollback-for:發生那些異常時回滾 
	          timeout:事務過期時間
	         -->
			<tx:method name="account*" propagation="REQUIRED"
				isolation="DEFAULT" read-only="false" rollback-for="" timeout="-1" />
		</tx:attributes>
	</tx:advice>

	<!-- 第三步:配置切面 切面即把增強用在方法的過程 -->
	<aop:config>
		<!-- 切入點 -->
		<aop:pointcut expression="execution(* cn.itcast.service.OrdersService.*(..))"
			id="pointcut1" />
		<!-- 切面 -->
		<aop:advisor advice-ref="txadvice" pointcut-ref="pointcut1" />
	</aop:config>


	<!-- 物件生成及屬性注入 -->
	<bean id="ordersService" class="cn.itcast.service.OrdersService">
		<property name="ordersDao" ref="ordersDao"></property>
	</bean>
	<bean id="ordersDao" class="cn.itcast.dao.OrdersDao">
		<property name="jdbcTemplate" ref="jdbcTemplate"></property>
	</bean>
	<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
		<property name="dataSource" ref="dataSource"></property>
	</bean>
複製程式碼

(3)基於註解的方式

OrdersService.java(業務邏輯層)

package cn.itcast.service;

import org.springframework.transaction.annotation.Isolation;
import org.springframework.transaction.annotation.Propagation;
import org.springframework.transaction.annotation.Transactional;

import cn.itcast.dao.OrdersDao;

@Transactional(propagation = Propagation.REQUIRED, isolation = Isolation.DEFAULT, readOnly = false, timeout = -1)
public class OrdersService {
	private OrdersDao ordersDao;

	public void setOrdersDao(OrdersDao ordersDao) {
		this.ordersDao = ordersDao;
	}

	// 呼叫dao的方法
	// 業務邏輯,寫轉賬業務
	public void accountMoney() {
		// 小馬多1000
		ordersDao.addMoney();
		// 加入出現異常如下面int i=10/0(銀行中可能為突然停電等。。。);結果:小馬賬戶多了1000而小王賬戶沒有少錢
		// 解決辦法是出現異常後進行事務回滾
		// int i = 10 / 0;// 事務管理配置後異常已經解決
		// 小王 少1000
		ordersDao.reduceMoney();
	}
}
複製程式碼

配置檔案:

	<!-- 配置c3po連線池 -->
	<bean id="dataSource" class="com.mchange.v2.c3p0.ComboPooledDataSource">
		<!-- 注入屬性值 -->
		<property name="driverClass" value="com.mysql.jdbc.Driver"></property>
		<property name="jdbcUrl" value="jdbc:mysql://localhost:3306/wangyiyun"></property>
		<property name="user" value="root"></property>
		<property name="password" value="153963"></property>
	</bean>
	<!-- 第一步:配置事務管理器 (和配置檔案方式一樣)-->
	<bean id="dataSourceTransactionManager"
		class="org.springframework.jdbc.datasource.DataSourceTransactionManager">
		<!-- 注入dataSource -->
		<property name="dataSource" ref="dataSource"></property>
	</bean>
	<!-- 第二步: 開啟事務註解 -->
	<tx:annotation-driven transaction-manager="dataSourceTransactionManager" />
	<!-- 第三步 在方法所在類上加註解 -->
	
	
	<!-- 物件生成及屬性注入 -->
	<bean id="ordersService" class="cn.itcast.service.OrdersService">
		<property name="ordersDao" ref="ordersDao"></property>
	</bean>
	<bean id="ordersDao" class="cn.itcast.dao.OrdersDao">
		<property name="jdbcTemplate" ref="jdbcTemplate"></property>
	</bean>
	<bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate">
		<property name="dataSource" ref="dataSource"></property>
	</bean>
複製程式碼

歡迎關注我的微信公眾號: "Java面試通關手冊" (一個有溫度的微信公眾號,期待與你共同進步~~~堅持原創,分享美文,分享各種Java學習資源):

相關文章