是否有任何简单的解决方案可以在新线程中使用JPA将数据保存到数据库中?
基于Spring的Web应用程序允许用户管理计划任务。在运行时,他可以创建和启动预定义任务的新实例。我正在使用spring的TaskScheduler,一切运行良好。
但是我需要将每个被解雇的任务的布尔结果保存到数据库中。我怎么能这样做?
编辑: 我必须概括我的问题:我需要从任务中调用@Service类的方法。因为在保存到数据库之前必须“处理”任务结果。
编辑2: 我的问题代码的简化版本来了。从调度程序调用saveTaskResult()时,将打印出消息,但不会将任何内容保存到db中。但每当我从控制器调用saveTaskResult()时,记录都会正确保存到数据库中。
@Service
public class DemoService {
@Autowired
private TaskResultDao taskResultDao;
@Autowired
private TaskScheduler scheduler;
public void scheduleNewTask() {
scheduler.scheduleWithFixedDelay(new Runnable() {
public void run() {
// do some action here
saveTaskResult(new TaskResult("result"));
}
}, 1000L);
}
@Transactional
public void saveTaskResult(TaskResult result) {
System.out.println("saving task result");
taskResultDao.persist(result);
}
}
答案 0 :(得分:22)
您的代码存在的问题是您希望在致电saveTaskResult()
时启动交易。这不会发生,因为Spring使用AOP来启动和停止事务。
如果从bean工厂获得事务性Spring bean的实例,或者通过依赖注入,你得到的实际上是围绕bean的代理。此代理在调用实际方法之前启动事务,并在方法完成后提交或回滚事务。
在这种情况下,您可以调用bean的本地方法,而无需通过事务代理。将saveTaskResult()
方法(用@Transactional
注释)放在另一个Spring bean中。将其他Spring bean注入DemoService,并从DemoService调用另一个Spring bean,一切都会好的。
答案 1 :(得分:3)
交易在线程本地存储中保存
如果您的其他方法正在运行带有@Transactional
注释的线程
默认设置为REQUIRED
,这意味着如果您从另一个线程运行一个带有@Transacitonal
注释的方法,您将拥有一个新事务(因为线程本地存储中没有保留事务)这个帖子)。