說說在 Spring 中,如何編程實現事務管理

Spring 爲編程式的事務管理,提供了相應的模板類 org.springframework.transaction.support.TransactionTemplate,能夠應對一些特殊場合的須要。mysql

TransactionTemplate 是線程安全的,因此能夠在多個類中共享 TransactionTemplate 實例,實現事務管理 。spring

TransactionTemplate 繼承了 DefaultTransactionDefinition,因此也擁有多種設置事務屬性的方法:sql

TransactionTemplate 類的主要方法:apache

方法 說明
public void setTransactionManager(PlatformTransactionManager transactionManager) 設置事務管理器。
public <T> T execute(TransactionCallback<T> action) 在 TransactionCallback 回調接口中,編寫須要以事務方式執行的數據訪問邏輯代碼。

TransactionCallback 接口只定義了一個方法 doInTransaction(TransactionStatus status)。Spring 還定義了一個抽象類 TransactionCallbackWithoutResult,用於只執行,但不須要返回結果的場景。編程

Spring 配置文件:安全

<?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:p="http://www.springframework.org/schema/p"
       xmlns:context="http://www.springframework.org/schema/context"
       xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans-4.0.xsd http://www.springframework.org/schema/context http://www.springframework.org/schema/context/spring-context-4.0.xsd">

    <!-- 掃描包含註解定義的類包-->
    <context:component-scan base-package="net.deniro.xxx.transaction"/>

    <!-- 數據源-->
    <bean id="dataSource" class="org.apache.commons.dbcp.BasicDataSource"
          destroy-method="close"
          p:driverClassName="com.mysql.jdbc.Driver"
          p:url="jdbc:mysql://127.0.0.1:3306/xxx"
          p:username="root"
          p:password="xxx"/>

    <!-- JDBC 模板-->
    <bean id="jdbcTemplate" class="org.springframework.jdbc.core.JdbcTemplate"
          p:dataSource-ref="dataSource"
            />

    <!-- 事務管理器-->
    <bean id="transactionManager"
          class="org.springframework.jdbc.datasource.DataSourceTransactionManager"
          p:dataSource-ref="dataSource"/>

    <!-- 事務模板 -->
    <bean id="TransactionTemplate"
          class="org.springframework.transaction.support.TransactionTemplate"
          p:transactionManager-ref="transactionManager"/>

</beans>
複製代碼

User 類:bash

public class User {
    private long id;
    private String name;

    public User(String name) {
        this.name = name;
    }

    public long getId() {
        return id;
    }

    public void setId(long id) {
        this.id = id;
    }

    public String getName() {
        return name;
    }

    public void setName(String name) {
        this.name = name;
    }


}
複製代碼

UserDao 類:ide

@Repository
public class UserDao {

    @Autowired
    private JdbcTemplate jdbcTemplate;


    /**
     * 保存帳號
     *
     * @param user
     * @throws SQLException
     */
    public void save(User user) {
        String sql = "insert into t_user(user_name) values(?)";
        jdbcTemplate.update(sql, user.getName());

        String sql2 = "insert into t_log(user_name) values(?)";
        jdbcTemplate.update(sql2, user.getName());
    }

}
複製代碼

爲了實驗事務是否生效,這裏的第一個 SQL 是正確的,而第二個 SQL 是錯誤的(找不到表)。單元測試

UserService 類:測試

@Service
public class UserService {

    @Autowired
    private UserDao userDao;


    @Autowired
    private TransactionTemplate template;


    public void save(final User user) {
        template.execute(new TransactionCallbackWithoutResult() {
            @Override
            protected void doInTransactionWithoutResult(TransactionStatus status) {
                userDao.save(user);
            }
        });
    }
}
複製代碼

單元測試:

public class UserServiceTest {

    ApplicationContext context;

    @BeforeMethod
    public void setUp() throws Exception {
        context = new ClassPathXmlApplicationContext("spring.xml");
    }

    @Test
    public void test() {
        UserService userService = (UserService) context.getBean("userService");
        final User user = new User("deniro");
        userService.save(user);
    }
}
複製代碼

爲了看到 Spring 的事務日誌,咱們能夠修改 log4j 事務配置,級別改成 DEBUG:

log4j.logger.org.springframework.jdbc.datasource.DataSourceTransactionManager=DEBUG
log4j.logger.org.springframework.transaction.interceptor=DEBUG
log4j.logger.org.springframework.transaction=DEBUG
複製代碼

運行結果分析:

Spring 首先建立一個新的事務。

在執行到第二個 SQL 語句時,拋出異常;Spring 執行回滾操做。說明事務生效咯O(∩_∩)O~

相關文章
相關標籤/搜索