PlatformTransactionManager platformTransactionManager = new DataSourceTransactionManager(dataSource);
步骤2:定义事务属性TransactionDefinition
定义事务属性,比如事务隔离级别、事务超时时间、事务传播方式、是否是只读事务等等。
spring中使用TransactionDefinition接口来表示事务的定义信息,有个子类比较常用:DefaultTransactionDefinition。
关于事务属性细节比较多,篇幅比较长,后面会专门有文章来详解。
步骤3:开启事务
调用事务管理器的getTransaction
方法,即可以开启一个事务
TransactionStatus transactionStatus = platformTransactionManager.getTransaction(transactionDefinition);
这个方法会返回一个TransactionStatus
表示事务状态的一个对象,通过TransactionStatus
提供的一些方法可以用来控制事务的一些状态,比如事务最终是需要回滚还是需要提交。
执行了getTransaction
后,spring内部会执行一些操作,为了方便大家理解,咱们看看伪代码:
//有一个全局共享的threadLocal对象 resources
static final ThreadLocal<Map<Object, Object>> resources = new NamedThreadLocal<>(“Transactional resources”);
//获取一个db的连接
DataSource datasource = platformTransactionManager.getDataSource();
Connection connection = datasource.getConnection();
//设置手动提交事务
connection.setAutoCommit(false);
Map<Object, Object> map = new HashMap<>();
map.put(datasource,connection);
resources.set(map);
上面代码,将数据源datasource和connection映射起来放在了ThreadLocal中,ThreadLocal大家应该比较熟悉,用于在同一个线程中共享数据;后面我们可以通过resources这个ThreadLocal获取datasource其对应的connection对象。
步骤4:执行业务操作
我们使用jdbcTemplate插入了2条记录。
jdbcTemplate.update(“insert into t_user (name) values (?)”, “test1-1”);
jdbcTemplate.update(“insert into t_user (name) values (?)”, “test1-2”);
大家看一下创建JdbcTemplate的代码,需要指定一个datasource
JdbcTemplate jdbcTemplate = new JdbcTemplate(dataSource);
再来看看创建事务管理器的代码
PlatformTransactionManager platformTransactionManager = new DataSourceTransactionManager(dataSource);
2者用到的是同一个dataSource,而事务管理器开启事务的时候,会创建一个连接,将datasource和connection映射之后丢在了ThreadLocal中,而JdbcTemplate内部执行db操作的时候,也需要获取连接,JdbcTemplate会以自己内部的datasource去上面的threadlocal中找有没有关联的连接,如果有直接拿来用,若没找到将重新创建一个连接,而此时是可以找到的,那么JdbcTemplate就参与到spring的事务中了。
步骤5:提交 or 回滚
//5.提交事务:platformTransactionManager.commit
platformTransactionManager.commit(transactionStatus);
//6.回滚事务:platformTransactionManager.rollback
platformTransactionManager.rollback(transactionStatus);
方式2:TransactionTemplate
方式1中部分代码是可以重用的,所以spring对其进行了优化,采用模板方法模式就其进行封装,主要省去了提交或者回滚事务的代码。
案例代码位置
测试代码
@Test
public void test1() throws Exception {
//定义一个数据源
org.apache.tomcat.jdbc.pool.DataSource dataSource = new org.apache.tomcat.jdbc.pool.DataSource();
dataSource.setDriverClassName(“com.mysql.jdbc.Driver”);
dataSource.setUrl(“jdbc:mysql://localhost:3306/javacode2018?characterEncoding=UTF-8”);
dataSource.setUsername(“root”);
dataSource.setPassword(“root123”);
dataSource.setInitialSize(5);
//定义一个JdbcTemplate,用来方便执行数据库增删改查
JdbcTemplate jdbcTempl