欧美bbbwbbbw肥妇,免费乱码人妻系列日韩,一级黄片

Spring源碼解析之編程式事務(wù)

 更新時(shí)間:2021年06月10日 16:58:54   作者:星夜孤帆  
今天給大家?guī)淼氖顷P(guān)于Java Spring的相關(guān)知識,文章圍繞著Spring編程式事務(wù)展開,文中有非常詳細(xì)的介紹,需要的朋友可以參考下

一、前言

在Spring中,事務(wù)有兩種實(shí)現(xiàn)方式:

編程式事務(wù)管理: 編程式事務(wù)管理使用TransactionTemplate可實(shí)現(xiàn)更細(xì)粒度的事務(wù)控制。聲明式事務(wù)管理: 基于Spring AOP實(shí)現(xiàn)。其本質(zhì)是對方法前后進(jìn)行攔截,然后在目標(biāo)方法開始之前創(chuàng)建或者加入一個(gè)事務(wù),在執(zhí)行完目標(biāo)方法之后根據(jù)執(zhí)行情況提交或者回滾事務(wù)。

聲明式事務(wù)管理不需要入侵代碼,通過@Transactional就可以進(jìn)行事務(wù)操作,更快捷而且簡單(尤其是配合spring boot自動(dòng)配置,可以說是精簡至極?。?,且大部分業(yè)務(wù)都可以滿足,推薦使用。

其實(shí)不管是編程式事務(wù)還是聲明式事務(wù),最終調(diào)用的底層核心代碼是一致的。本文主要介紹編程式事務(wù)的一些應(yīng)用,以及獨(dú)有的源碼分析,再在其他文章中進(jìn)入核心源碼貫穿式講解。

二、編程式事務(wù)解析

編程式事務(wù),Spring已經(jīng)給我們提供好了模板類TransactionTemplate,可以很方便的使用,如下圖:

TransactionTemplate全路徑名是:org.springframework.transaction.support.TransactionTemplate??窗仓懒诉@是spring對事務(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()異常回滾:調(diào)用AbstractPlatformTransactionManager的rollback(),事務(wù)提交回滾。源碼見后續(xù)文章

三、編程式事務(wù)示例

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),回滾的化,不會回滾保存點(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;
            }
        });
    }
}

四、TransactionCallback

編程式事務(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;
        });
    }
 
}

五、TransactionCallbackWithoutResult

編程式事務(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方式,可以正常回滾,不知道什么原因

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);
    }
}

參考文章參考文章

到此這篇關(guān)于Spring源碼解析之編程式事務(wù)的文章就介紹到這了,更多相關(guān)Spring編程式事務(wù)內(nèi)容請搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!

相關(guān)文章

  • 如何使用IDEA查看java文件編譯后的字節(jié)碼內(nèi)容

    如何使用IDEA查看java文件編譯后的字節(jié)碼內(nèi)容

    這篇文章主要介紹了如何使用IDEA查看java文件編譯后的字節(jié)碼內(nèi)容,具有很好的參考價(jià)值,希望對大家有所幫助。如有錯(cuò)誤或未考慮完全的地方,望不吝賜教
    2022-03-03
  • Java中Cglib代理和JDK代理的區(qū)別詳解

    Java中Cglib代理和JDK代理的區(qū)別詳解

    這篇文章主要介紹了Java中Cglib代理和JDK代理的區(qū)別詳解,Cglib代理功能更強(qiáng),無論目標(biāo)類是否實(shí)現(xiàn)了接口都可以代理,他是基于繼承的方式來代理目標(biāo)類,如果目標(biāo)類也實(shí)現(xiàn)了接口,代理類也會實(shí)現(xiàn)一次,需要的朋友可以參考下
    2023-09-09
  • Java源碼解析HashMap的keySet()方法

    Java源碼解析HashMap的keySet()方法

    今天小編就為大家分享一篇關(guān)于Java源碼解析HashMap的keySet()方法,小編覺得內(nèi)容挺不錯(cuò)的,現(xiàn)在分享給大家,具有很好的參考價(jià)值,需要的朋友一起跟隨小編來看看吧
    2019-01-01
  • SpringBoot Admin的簡單使用的方法步驟

    SpringBoot Admin的簡單使用的方法步驟

    本文主要介紹了SpringBoot Admin的簡單使用的方法步驟,文中通過示例代碼介紹的非常詳細(xì),具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2022-01-01
  • springboot內(nèi)嵌Tomcat安全漏洞修復(fù)方式

    springboot內(nèi)嵌Tomcat安全漏洞修復(fù)方式

    針對CVE-2020-1938漏洞,建議升級Tomcat至安全版本以避免受影響,影響版本包括:Apache Tomcat 9.x小于9.0.31、Apache Tomcat 8.x小于8.5.51、Apache Tomcat 7.x小于7.0.100及Apache Tomcat 6.x,
    2024-10-10
  • Java創(chuàng)建,編輯與刪除Excel迷你圖表的實(shí)現(xiàn)方法

    Java創(chuàng)建,編輯與刪除Excel迷你圖表的實(shí)現(xiàn)方法

    迷你圖是Excel工作表單元格中表示數(shù)據(jù)的微型圖表。本文將通過Java代碼示例介紹如何在Excel中創(chuàng)建迷你圖表,以及編輯和刪除表格中的迷你圖表,需要的可以參考一下
    2022-05-05
  • java編程之單元測試(Junit)實(shí)例分析(附實(shí)例源碼)

    java編程之單元測試(Junit)實(shí)例分析(附實(shí)例源碼)

    這篇文章主要介紹了java編程之單元測試(Junit),結(jié)合實(shí)例形式較為詳細(xì)的分析總結(jié)了Java單元測試的原理、步驟及相關(guān)注意事項(xiàng),并附帶了完整代碼供讀者下載參考,需要的朋友可以參考下
    2015-11-11
  • 徹底搞定堆排序:二叉堆

    徹底搞定堆排序:二叉堆

    二叉堆有兩種:最大堆和最小堆。最大堆:父結(jié)點(diǎn)的鍵值總是大于或等于任何一個(gè)子節(jié)點(diǎn)的鍵值;最小堆:父結(jié)點(diǎn)的鍵值總是小于或等于任何一個(gè)子節(jié)點(diǎn)的鍵值
    2021-07-07
  • Java Web使用Html5 FormData實(shí)現(xiàn)多文件上傳功能

    Java Web使用Html5 FormData實(shí)現(xiàn)多文件上傳功能

    這篇文章主要介紹了Java Web使用Html5 FormData實(shí)現(xiàn)多文件上傳功能,需要的朋友可以參考下
    2017-07-07
  • 詳解Java中Vector和ArrayList的區(qū)別

    詳解Java中Vector和ArrayList的區(qū)別

    這篇文章主要為大家詳細(xì)介紹了Java中Vector和ArrayList的區(qū)別,具有一定的參考價(jià)值,感興趣的小伙伴們可以參考一下
    2016-10-10

最新評論