您好,登錄后才能下訂單哦!
這篇文章主要為大家展示了“Spring源碼解析之編程式事務(wù)的示例分析”,內(nèi)容簡(jiǎn)而易懂,條理清晰,希望能夠幫助大家解決疑惑,下面讓小編帶領(lǐng)大家一起研究并學(xué)習(xí)一下“Spring源碼解析之編程式事務(wù)的示例分析”這篇文章吧。
在Spring中,事務(wù)有兩種實(shí)現(xiàn)方式:
編程式事務(wù)管理: 編程式事務(wù)管理使用TransactionTemplate可實(shí)現(xiàn)更細(xì)粒度的事務(wù)控制。聲明式事務(wù)管理: 基于Spring AOP實(shí)現(xiàn)。其本質(zhì)是對(duì)方法前后進(jìn)行攔截,然后在目標(biāo)方法開始之前創(chuàng)建或者加入一個(gè)事務(wù),在執(zhí)行完目標(biāo)方法之后根據(jù)執(zhí)行情況提交或者回滾事務(wù)。聲明式事務(wù)管理不需要入侵代碼,通過@Transactional就可以進(jìn)行事務(wù)操作,更快捷而且簡(jiǎn)單(尤其是配合spring boot自動(dòng)配置,可以說是精簡(jiǎn)至極?。?,且大部分業(yè)務(wù)都可以滿足,推薦使用。
其實(shí)不管是編程式事務(wù)還是聲明式事務(wù),最終調(diào)用的底層核心代碼是一致的。本文主要介紹編程式事務(wù)的一些應(yīng)用,以及獨(dú)有的源碼分析,再在其他文章中進(jìn)入核心源碼貫穿式講解。
編程式事務(wù),Spring已經(jīng)給我們提供好了模板類TransactionTemplate,可以很方便的使用,如下圖:
TransactionTemplate全路徑名是:org.springframework.transaction.support.TransactionTemplate??窗仓懒诉@是spring對(duì)事務(wù)的模板類。(spring動(dòng)不動(dòng)就是各種Template...),看下類圖先:
實(shí)現(xiàn)了TransactionOperations、InitializingBean這2個(gè)接口(熟悉spring源碼的知道這個(gè)InitializingBean又是老套路),我們來看下接口源碼如下:
public interface TransactionOperations { @Nullable <T> T execute(TransactionCallback<T> action) throws TransactionException; } public interface InitializingBean { void afterPropertiesSet() throws Exception; }
如上圖,TransactionOperations這個(gè)接口用來執(zhí)行事務(wù)的回調(diào)方法,InitializingBean這個(gè)是典型的spring bean初始化流程中的預(yù)留接口,專用用來在bean屬性加載完畢時(shí)執(zhí)行的方法。
回到正題,TransactionCallback和TransactionCallbackWithoutResult做了什么
@FunctionalInterface public interface TransactionCallback<T> { @Nullable T doInTransaction(TransactionStatus status); } public abstract class TransactionCallbackWithoutResult implements TransactionCallback<Object> { @Override @Nullable public final Object doInTransaction(TransactionStatus status) { doInTransactionWithoutResult(status); return null; } protected abstract void doInTransactionWithoutResult(TransactionStatus status); }
可見TransactionCallbackWithResult實(shí)現(xiàn)了TransactionCallback接口,重寫了doIntransaction方法,在內(nèi)部調(diào)用了doInTransactionWithoutResult方法,幫我們返回了null,所以,我們就不需要再指定返回值了。
TransactionTemplate的2個(gè)接口的impl方法做了什么?
@Override public void afterPropertiesSet() { if (this.transactionManager == null) { throw new IllegalArgumentException("Property 'transactionManager' is required"); } } @Override public <T> T execute(TransactionCallback<T> action) throws TransactionException { // 內(nèi)部封裝好的事務(wù)管理器 if (this.transactionManager instanceof CallbackPreferringPlatformTransactionManager) { return ((CallbackPreferringPlatformTransactionManager) this.transactionManager).execute(this, action); }// 需要手動(dòng)獲取事務(wù),執(zhí)行方法,提交事務(wù)的管理器 else {// 1.獲取事務(wù)狀態(tài) TransactionStatus status = this.transactionManager.getTransaction(this); T result; try {// 2.執(zhí)行業(yè)務(wù)邏輯 result = action.doInTransaction(status); } catch (RuntimeException ex) { // 應(yīng)用運(yùn)行時(shí)異常 -> 回滾 rollbackOnException(status, ex); throw ex; } catch (Error err) { // Error異常 -> 回滾 rollbackOnException(status, err); throw err; } catch (Throwable ex) { // 未知異常 -> 回滾 rollbackOnException(status, ex); throw new UndeclaredThrowableException(ex, "TransactionCallback threw undeclared checked exception"); }// 3.事務(wù)提交 this.transactionManager.commit(status); return result; } }
如上圖所示,實(shí)際上afterPropertiesSet只是校驗(yàn)了事務(wù)管理器不為空,execute()才是核心方法,execute主要步驟:
1.getTransaction()獲取事務(wù)
2.doInTransaction()執(zhí)行業(yè)務(wù)邏輯,這里就是用戶自定義的業(yè)務(wù)代碼。如果是沒有返回值的,就是doInTransactionWithoutResult()。
3.commit()事務(wù)提交:調(diào)用AbstractPlatformTransactionManager的commit,rollbackOnException()異?;貪L:調(diào)用AbstractPlatformTransactionManager的rollback(),事務(wù)提交回滾。源碼見后續(xù)文章
public class SpringTransactionExample { private static String url = "jdbc:mysql://localhost:3306/spring_transaction?useSSL=false&characterEncoding=utf-8&autoReconnect=true"; private static String user = "root"; private static String password = "root"; public static void main(String[] args) { // 獲取數(shù)據(jù)源 final DataSource ds = new DriverManagerDataSource(url, user, password); // 編程式事務(wù) final TransactionTemplate template = new TransactionTemplate(); // 設(shè)置事務(wù)管理器 template.setTransactionManager(new DataSourceTransactionManager(ds)); template.execute(new TransactionCallback<Object>() { @Override public Object doInTransaction(TransactionStatus transactionStatus) { Connection conn = DataSourceUtils.getConnection(ds); Object savePoint = null; try { { // 插入 PreparedStatement prepare = conn.prepareStatement("insert into person(id, name) values (?, ?)"); prepare.setString(1, "1"); prepare.setString(2, "1111"); prepare.executeUpdate(); } // 設(shè)置保存點(diǎn),回滾的化,不會(huì)回滾保存點(diǎn)之前的操作 savePoint = transactionStatus.createSavepoint(); { // 插入 PreparedStatement prepare = conn.prepareStatement("insert into person(id, name) values (?, ?)"); prepare.setString(1, "2"); prepare.setString(2, "222"); prepare.executeUpdate(); } { // 更新 PreparedStatement prepare = conn.prepareStatement("update person set name = ? where id = ?"); prepare.setString(1, "jak"); prepare.setInt(2, 6); prepare.executeUpdate(); // 模擬異常 // int i = 1 / 0; } } catch (SQLException e) { e.printStackTrace(); } catch (Exception e) { System.out.println("更新失敗"); if (savePoint != null) { // 回滾到保存點(diǎn) transactionStatus.rollbackToSavepoint(savePoint); } else { transactionStatus.setRollbackOnly(); } } return null; } }); } }
編程式事務(wù)帶返回值
public class TransactionCallBackTest { private static String url = "jdbc:mysql://localhost:3306/spring_transaction?useSSL=false&characterEncoding=utf-8&autoReconnect=true"; private static String user = "root"; private static String password = "root"; public static void main(String[] args) { // 獲取數(shù)據(jù)源 final DataSource ds = new DriverManagerDataSource(url, user, password); // 編程式事務(wù) final TransactionTemplate template = new TransactionTemplate(); // 設(shè)置事務(wù)管理器 template.setTransactionManager(new DataSourceTransactionManager(ds)); Connection connection = DataSourceUtils.getConnection(ds); test1(template, connection); test2(template, connection); } // 方式一: 匿名內(nèi)部類 @SuppressWarnings("all") public static void test1(TransactionTemplate template, Connection connection) { // TransactionCallback有返回值 template.execute(new TransactionCallback<Object>() { @Override public Object doInTransaction(TransactionStatus status) { try { // 插入 PreparedStatement prepare = connection.prepareStatement("insert into person(id, name) values (?, ?)"); prepare.setInt(1, 1); prepare.setString(2, "jak"); prepare.executeUpdate(); // 模擬異常 // int i = 1 / 0; System.out.println("數(shù)據(jù)已插入"); } catch (SQLException e) { e.printStackTrace(); } catch (Exception e) { System.out.println("更新失敗"); status.setRollbackOnly(); } return null; } }); } // 方式二:lamda表達(dá)式 @SuppressWarnings("all") public static void test2(TransactionTemplate template, Connection connection) { template.execute((status) -> { try { // 插入 PreparedStatement prepare = connection.prepareStatement("insert into person(id, name) values (?, ?)"); prepare.setInt(1, 2); prepare.setString(2, "hyd"); prepare.executeUpdate(); // 模擬異常 // int i = 1 / 0; System.out.println("數(shù)據(jù)已插入"); } catch (SQLException e) { e.printStackTrace(); } catch (Exception e) { System.out.println("更新失敗"); status.setRollbackOnly(); } return null; }); } }
編程式事務(wù)不帶返回值
public class TransactionCallbackWithoutResultTest { private static String url = "jdbc:mysql://localhost:3306/spring_transaction?useSSL=false&characterEncoding=utf-8&autoReconnect=true"; private static String user = "root"; private static String password = "root"; @SuppressWarnings("all") public static void test(TransactionTemplate template, Connection connection) { template.execute(new TransactionCallbackWithoutResult() { // doInTransactionWithoutResult無返回值 @Override public void doInTransactionWithoutResult(TransactionStatus status) { try { // 插入 PreparedStatement prepare = connection.prepareStatement("insert into person(id, name) values (?, ?)"); prepare.setInt(1, 1); prepare.setString(2, "jak"); prepare.executeUpdate(); // 模擬異常 // int i = 1 / 0; System.out.println("數(shù)據(jù)已插入"); } catch (SQLException e) { e.printStackTrace(); } catch (Exception e) { System.out.println("更新失敗"); status.setRollbackOnly(); } } }); } public static void main(String[] args) { // 獲取數(shù)據(jù)源 final DataSource ds = new DriverManagerDataSource(url, user, password); // 編程式事務(wù) final TransactionTemplate template = new TransactionTemplate(); // 設(shè)置事務(wù)管理器 template.setTransactionManager(new DataSourceTransactionManager(ds)); Connection connection = DataSourceUtils.getConnection(ds); test(template, connection); } }
踩坑指南,上述方式 不知道為啥,事務(wù)一直不回滾,改為jdbcTemplate方式,可以正?;貪L,不知道什么原因
public class jdbcTemplateTest { private static String url = "jdbc:mysql://localhost:3306/spring_transaction?useSSL=false&characterEncoding=utf-8&autoReconnect=true"; private static String user = "root"; private static String password = "root"; @SuppressWarnings("all") public static void test(TransactionTemplate template, JdbcTemplate jdbcTemplate) { template.execute(new TransactionCallbackWithoutResult() { // doInTransactionWithoutResult無返回值 @SneakyThrows @Override public void doInTransactionWithoutResult(TransactionStatus status) { try { // 插入 jdbcTemplate.execute("insert into person(id, name) values (2, 'jak')"); // 模擬異常 int i = 1 / 0; System.out.println("數(shù)據(jù)已插入"); } catch (Exception e) { // 標(biāo)記事務(wù)回滾 status.setRollbackOnly(); } } }); } public static void main(String[] args) { // 獲取數(shù)據(jù)源 final DataSource ds = new DriverManagerDataSource(url, user, password); // 編程式事務(wù) final TransactionTemplate template = new TransactionTemplate(); // jdbcTemplate final JdbcTemplate jdbcTemplate = new JdbcTemplate(); // 設(shè)置事務(wù)管理器 template.setTransactionManager(new DataSourceTransactionManager(ds)); // 配置數(shù)據(jù)源 jdbcTemplate.setDataSource(ds); test(template, jdbcTemplate); } }
以上是“Spring源碼解析之編程式事務(wù)的示例分析”這篇文章的所有內(nèi)容,感謝各位的閱讀!相信大家都有了一定的了解,希望分享的內(nèi)容對(duì)大家有所幫助,如果還想學(xué)習(xí)更多知識(shí),歡迎關(guān)注億速云行業(yè)資訊頻道!
免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。