溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

Spring源碼解析之編程式事務(wù)的示例分析

發(fā)布時(shí)間:2021-06-11 09:15:57 來源:億速云 閱讀:162 作者:小新 欄目:開發(fā)技術(shù)

這篇文章主要為大家展示了“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ù)解析

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

Spring源碼解析之編程式事務(wù)的示例分析

TransactionTemplate全路徑名是:org.springframework.transaction.support.TransactionTemplate??窗仓懒诉@是spring對(duì)事務(wù)的模板類。(spring動(dòng)不動(dòng)就是各種Template...),看下類圖先:

Spring源碼解析之編程式事務(wù)的示例分析

實(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ù)文章

三、編程式事務(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),回滾的化,不會(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;
            }
        });
    }
}

四、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方式,可以正?;貪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è)資訊頻道!

向AI問一下細(xì)節(jié)

免責(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)容。

AI