@Async 正在终止 Hibernate 事务
@Async is killing Hibernate transaction
我正在为我的 REST api 使用 Open-Session-In-View 事务模型,如下所示:
@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
sessionFactory.getCurrentSession().beginTransaction();
chain.doFilter(request, response);
sessionFactory.getCurrentSession().getTransaction().commit();
}
这项工作很好。我想添加 @Async 功能。所以我创建了:
@Configuration
@EnableAsync
public class AsyncConfig implements AsyncConfigurer {
@Override
@Bean(destroyMethod="shutdown")
public Executor getAsyncExecutor() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
executor.setCorePoolSize(18);
executor.setMaxPoolSize(18);
executor.initialize();
executor.setDaemon(true);
executor.setWaitForTasksToCompleteOnShutdown(false);
return executor;
}
@Override
public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() {
return new SimpleAsyncUncaughtExceptionHandler();
}
}
和:
@Component
public class AsyncMarketCaller {
@Inject
MarketManagerFactory marketManagerFactory;
@Async
public Future<List<Product>> getProducts(){
MarketManager productManager = marketManagerFactory.obtainMarketManager(market);
result = productManager.getProducts();
....
}
}
productManager 调用另一个@Service
@Service
public class DefaultIdentifierManager implements IdentifierManager{
@Inject
UpcEanDAO upcEanDAO;
@Override
public String getTitleForIdentifier(String identifier){
UpcEan upcEan = upcEanDAO.find(identifier);
}
}
然而,对于 upcEanDAO.find(identifier)
我得到一个例外:
Caused by: org.hibernate.HibernateException: get is not valid without active transaction
在我添加 @Async
对 getProducts()
进行异步调用的能力之前,它工作得很好,所以我假设 @Async
终止了我用 Hibernate 打开的事务。
我尝试根据这里的另一个答案添加 @Transactional
到用 @Async
注释的方法,但它没有帮助。
有什么想法吗?
已编辑
我编辑了代码
@Component
public class AsyncMarketCaller {
@Inject
AsyncMarketService asyncMarketService;
@Async
public Future<List<Product>> getProducts(){
asyncMarketService.getProducts();
}
}
和
@Service
public class AsyncMarketService {
@Inject
MarketManagerFactory marketManagerFactory;
@Transactional
public Future<List<Product>> getProducts()
....
}
}
我在日志里看到了
50689 DEBUG [localhost-startStop-1] org.springframework.transaction.annotation.AnnotationTransactionAttributeSource - Adding transactional method 'AsyncMarketService.getProducts' with attribute: PROPAGATION_REQUIRED,ISOLATION_DEFAULT; ''
但这无济于事。请注意,我的方法 AsyncMarketService.getProducts 不会直接调用数据库,它会调用其他方法,并且只有其中一个方法会进行调用。
我还在上面添加了实际调用 DB 的那个:@Transactional
49992 DEBUG [localhost-startStop-1] org.springframework.transaction.annotation.AnnotationTransactionAttributeSource - Adding transactional method 'DefaultIdentifierManager.getTitleForIdentifier' with attribute: PROPAGATION_REQUIRED,ISOLATION_DEFAULT; '
'
Hibernate 事务在 ThreadLocal
基础上工作。
当您将另一个线程与 @Async
一起使用时,将不会有任何事务处于活动状态。
您可以通过让异步方法调用另一个由 @Transactional
注释的 bean 来实现此功能。
这里我稍微解释一下这个方法:
How do I properly do a background thread when using Spring Data and Hibernate?
我正在为我的 REST api 使用 Open-Session-In-View 事务模型,如下所示:
@Override
public void doFilter(ServletRequest request, ServletResponse response, FilterChain chain)
sessionFactory.getCurrentSession().beginTransaction();
chain.doFilter(request, response);
sessionFactory.getCurrentSession().getTransaction().commit();
}
这项工作很好。我想添加 @Async 功能。所以我创建了:
@Configuration
@EnableAsync
public class AsyncConfig implements AsyncConfigurer {
@Override
@Bean(destroyMethod="shutdown")
public Executor getAsyncExecutor() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
executor.setCorePoolSize(18);
executor.setMaxPoolSize(18);
executor.initialize();
executor.setDaemon(true);
executor.setWaitForTasksToCompleteOnShutdown(false);
return executor;
}
@Override
public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() {
return new SimpleAsyncUncaughtExceptionHandler();
}
}
和:
@Component
public class AsyncMarketCaller {
@Inject
MarketManagerFactory marketManagerFactory;
@Async
public Future<List<Product>> getProducts(){
MarketManager productManager = marketManagerFactory.obtainMarketManager(market);
result = productManager.getProducts();
....
}
}
productManager 调用另一个@Service
@Service
public class DefaultIdentifierManager implements IdentifierManager{
@Inject
UpcEanDAO upcEanDAO;
@Override
public String getTitleForIdentifier(String identifier){
UpcEan upcEan = upcEanDAO.find(identifier);
}
}
然而,对于 upcEanDAO.find(identifier)
我得到一个例外:
Caused by: org.hibernate.HibernateException: get is not valid without active transaction
在我添加 @Async
对 getProducts()
进行异步调用的能力之前,它工作得很好,所以我假设 @Async
终止了我用 Hibernate 打开的事务。
我尝试根据这里的另一个答案添加 @Transactional
到用 @Async
注释的方法,但它没有帮助。
有什么想法吗?
已编辑
我编辑了代码
@Component
public class AsyncMarketCaller {
@Inject
AsyncMarketService asyncMarketService;
@Async
public Future<List<Product>> getProducts(){
asyncMarketService.getProducts();
}
}
和
@Service
public class AsyncMarketService {
@Inject
MarketManagerFactory marketManagerFactory;
@Transactional
public Future<List<Product>> getProducts()
....
}
}
我在日志里看到了
50689 DEBUG [localhost-startStop-1] org.springframework.transaction.annotation.AnnotationTransactionAttributeSource - Adding transactional method 'AsyncMarketService.getProducts' with attribute: PROPAGATION_REQUIRED,ISOLATION_DEFAULT; ''
但这无济于事。请注意,我的方法 AsyncMarketService.getProducts 不会直接调用数据库,它会调用其他方法,并且只有其中一个方法会进行调用。
我还在上面添加了实际调用 DB 的那个:@Transactional
49992 DEBUG [localhost-startStop-1] org.springframework.transaction.annotation.AnnotationTransactionAttributeSource - Adding transactional method 'DefaultIdentifierManager.getTitleForIdentifier' with attribute: PROPAGATION_REQUIRED,ISOLATION_DEFAULT; '
'
Hibernate 事务在 ThreadLocal
基础上工作。
当您将另一个线程与 @Async
一起使用时,将不会有任何事务处于活动状态。
您可以通过让异步方法调用另一个由 @Transactional
注释的 bean 来实现此功能。
这里我稍微解释一下这个方法: How do I properly do a background thread when using Spring Data and Hibernate?