MHo*_*gge 1 java postgresql transactions spring-boot
我尝试执行事务操作并故意抛出异常,以验证回滚是否已完成,但回滚并未执行。
PostgreSQL 数据库版本为 12.1-1,基于 Docker。
这是包含注释的服务@Transactional:
@Service
public class MyTestService {
@Autowired
private DocumentDataDao documentDataDao;
@Transactional
public void test() {
DocumentData data = new DocumentData();
data.setData(UUID.randomUUID().toString());
documentDataDao.create(data);
throw new IllegalArgumentException("Test rollback");
}
}
Run Code Online (Sandbox Code Playgroud)
该create函数使用 aNamedParameterJdbcTemplate插入数据:
String statement = String.format("INSERT INTO %s (%s) VALUES (%s) RETURNING %s", tableName,
String.join(",", insertingColumns), String.join(",", values),
String.join(",", returningColumns));
return getNamedJdbcTemplate().queryForObject(statement, parameters, getRowMapper());
Run Code Online (Sandbox Code Playgroud)
该test函数是从另一个服务调用的:
@Service
public class ApplicationStartupListener {
private Logger log = LoggerFactory.getLogger(ApplicationStartupListener.class);
@Autowired
private MyTestService testService;
@PostConstruct
public void init() {
try {
testService.test();
} catch (Exception e) {
log.error("fail to start", e);
}
}
}
Run Code Online (Sandbox Code Playgroud)
调试的时候发现如果没有执行回滚是因为事务正在执行IDLE。
以下是未执行的rollback函数:PgConnectionexecuteTransactionCommand
public void rollback() throws SQLException {
checkClosed();
if (autoCommit) {
throw new PSQLException(GT.tr("Cannot rollback when autoCommit is enabled."),
PSQLState.NO_ACTIVE_SQL_TRANSACTION);
}
if (queryExecutor.getTransactionState() != TransactionState.IDLE) {
executeTransactionCommand(rollbackQuery);
}
}
Run Code Online (Sandbox Code Playgroud)
任何关于为什么事务被标记为空闲并停止执行回滚方法的提示将不胜感激。
正如@M。Deinum 提到,不能保证使用时已创建事务代理@PostConstruct。这就是为什么我用以下方法进行测试ApplicationRunner:
@Component
public class AppStartupRunner implements ApplicationRunner {
@Autowired
private MyTestService testService;
@Override
public void run(ApplicationArguments args) throws Exception {
testService.test();
}
}
Run Code Online (Sandbox Code Playgroud)
但这也不起作用。
我还尝试test在应用程序启动后不久通过使用 aRestController并向其发送 HTTP 请求来运行该方法,但仍然存在同样的问题。
@RestController
public class AppController {
@Autowired
private MyTestService testService;
@GetMapping("/test")
public ResponseEntity<Object> test() {
testService.test();
return ResponseEntity.ok().build();
}
}
Run Code Online (Sandbox Code Playgroud)
我将 PostgreSQL JDBC 版本从 42.2.2 升级到 42.2.18(目前最新),但在尝试回滚时连接仍然处于 IDLE 状态。
我在 git 存储库中重现了该问题:https://github.com/Martin-Hogge/spring-boot-postgresql-transactional-example/tree/master。
我检查了您想要在单个应用程序中使用多个架构(数据源、jdbc 模板)的架构。@Transactional仅管理名为 的应用程序默认数据源HikariPool-1。当您调用其余方法时,将创建名为 的新 hikari 池HikariPool-2。您的操作正在进行HikariPool-2,但@Transactional仅进行管理HikariPool-1。
@Transactional的事务管理器参数不能动态更改。因此,您可以定义一个新的自定义注释来管理您的事务。或者您可以使用TransactionTemplate注释来代替。
我创建了一个简单的自定义事务管理方面。它与定义的 dao 的数据源和事务生命周期一起工作。
@Service
public class MyTestService {
@Autowired
private DocumentDataDao documentDataDao;
@CustomTransactional(DocumentDataDao.class)
public void test() {
DocumentData data = new DocumentData();
data.setData(UUID.randomUUID().toString());
documentDataDao.create(data);
throw new IllegalArgumentException("Test rollback");
}
}
Run Code Online (Sandbox Code Playgroud)
package com.example.transactional;
import java.lang.annotation.*;
@Target({ElementType.METHOD, ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Inherited
@Documented
public @interface CustomTransactional {
Class<? extends BaseDao<?>> value();
}
Run Code Online (Sandbox Code Playgroud)
package com.example.transactional;
import org.aspectj.lang.ProceedingJoinPoint;
import org.aspectj.lang.annotation.Around;
import org.aspectj.lang.annotation.Aspect;
import org.springframework.beans.BeansException;
import org.springframework.context.ApplicationContext;
import org.springframework.context.ApplicationContextAware;
import org.springframework.stereotype.Component;
import java.util.HashMap;
import java.util.Map;
@Aspect
@Component
public class CustomTransactionalAspect implements ApplicationContextAware {
private ApplicationContext applicationContext;
private Map<Class<? extends BaseDao<?>>, BaseDao<?>> classMap = new HashMap<>();
@Around("@annotation(customTransactional)")
public Object customTransaction(ProceedingJoinPoint joinPoint, CustomTransactional customTransactional) throws Throwable {
BaseDao<?> baseDao = getBaseDao(customTransactional.value());
// custom transaction management
return baseDao.getConnectionHandler().getTransactionTemplate().execute(status -> {
try {
return joinPoint.proceed();
} catch (Throwable throwable) {
throw new RuntimeException(throwable);
}
});
}
/**
* Search {@link BaseDao} class on spring beans
*
* @param clazz Target dao class type
* @return Spring bean object
*/
private BaseDao<?> getBaseDao(Class<? extends BaseDao<?>> clazz) {
return classMap.computeIfAbsent(clazz, c -> applicationContext.getBean(c));
}
@Override
public void setApplicationContext(ApplicationContext applicationContext) throws BeansException {
this.applicationContext = applicationContext;
}
}
Run Code Online (Sandbox Code Playgroud)
我添加的transactionTemplate是交易操作
public class ConnectionHandler {
private NamedParameterJdbcTemplate namedJdbcTemplate;
private JdbcTemplate jdbcTemplate;
private TransactionTemplate transactionTemplate;
private String schema;
public ConnectionHandler(DataSource dataSource, String schema) {
this.namedJdbcTemplate = new NamedParameterJdbcTemplate(dataSource);
this.jdbcTemplate = new JdbcTemplate(dataSource);
this.schema = schema;
this.transactionTemplate = new TransactionTemplate(new DataSourceTransactionManager(dataSource));
}
public NamedParameterJdbcTemplate getNamedJdbcTemplate() {
return namedJdbcTemplate;
}
public JdbcTemplate getJdbcTemplate() {
return jdbcTemplate;
}
public String getSchema() {
return schema;
}
public TransactionTemplate getTransactionTemplate() {
return transactionTemplate;
}
}
Run Code Online (Sandbox Code Playgroud)
将修饰符 更改getConnectionHandler为public。
public ConnectionHandler getConnectionHandler() {
return getDataSource().getConnection(getSchemaName());
}
Run Code Online (Sandbox Code Playgroud)
您可以删除postgresql.version,spring-jdbc.version和HikariCP.version。问题与版本无关。添加spring-boot-starter-aop方面操作的依赖项。
<dependencies>
...
<dependency>
<groupId>org.springframework.boot</groupId>
<artifactId>spring-boot-starter-aop</artifactId>
</dependency>
</dependencies>
Run Code Online (Sandbox Code Playgroud)
| 归档时间: |
|
| 查看次数: |
2801 次 |
| 最近记录: |