@Scheduled + Hibernate -> LazyInitializationException
@Scheduled + Hibernate -> LazyInitializationException
我在 Spring Boot 2.0.5 下,使用 Spring Data JPA
我有一个class比如这个(方便理解):
@Component
public class Synchronizer {
@Autowired
private MyService myService;
@Transactional
public void synchronizeAuto() {
List<MyTest> tests = myService.getTests();
tests.get(0).getMyLazyObject().getName();
}
}
配置在这里(还有其他我省略的配置文件):
@Configuration
@EnableAsync
@EnableScheduling
@EnableTransactionManagement
public class SpringAsyncConfiguration implements AsyncConfigurer, SchedulingConfigurer {
@Autowired
private AppConfigProperties appConfigProperties;
@Autowired
private Synchronizer synchronizer;
@Override
public Executor getAsyncExecutor() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
executor.setCorePoolSize(appConfigProperties.getThreadpoolCorePoolSize());
executor.setMaxPoolSize(appConfigProperties.getThreadpoolMaxPoolSize());
executor.setQueueCapacity(appConfigProperties.getThreadpoolQueueCapacity());
executor.setThreadNamePrefix("threadPoolExecutor-");
executor.initialize();
return executor;
}
@Override
public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() {
return new AsyncExceptionHandler();
}
@Override
public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {
taskRegistrar.addCronTask(new Runnable() {
@Override
@Transactional
public void run() {
synchronizer.synchronizeAuto();
}
}, appConfigProperties.getCronExpression());
}
}
class MyService 调用 Spring JPA 存储库以获取所有 "Test" 个实例
一个 "Test" 实例有延迟加载 (MyLazyObject)
无论如何,如果我从我的控制器调用方法,一切都会像魅力一样。
当它来自调度程序 运行 时,我收到以下错误:
org.hibernate.LazyInitializationException: failed to lazily initialize a collection of role: com.tinea.apix.service.model.entity.apim.APIManagerEntity.synchroHistory, could not initialize proxy - no Session
有什么想法吗?
由于使用了在配置时调用的configureTasks
,所以很早就创建了Syncronizer
。太早了,它不再符合代理 creation/post 处理的条件。这反过来至少导致任务使用未代理的实例并且没有应用 @Transactional
。
相反,您应该使用 @Scheduled
注释和 cronString
属性 来像现在一样解决它。
@Scheduled(cron="@appConfigProperties.cronExpression")
SpEL 表达式中的 @
符号表示应解析具有给定名称的 bean。
@Scheduled(cron = "0 30 17 * * ?")
否则我们也可以这样使用
我在 Spring Boot 2.0.5 下,使用 Spring Data JPA
我有一个class比如这个(方便理解):
@Component
public class Synchronizer {
@Autowired
private MyService myService;
@Transactional
public void synchronizeAuto() {
List<MyTest> tests = myService.getTests();
tests.get(0).getMyLazyObject().getName();
}
}
配置在这里(还有其他我省略的配置文件):
@Configuration
@EnableAsync
@EnableScheduling
@EnableTransactionManagement
public class SpringAsyncConfiguration implements AsyncConfigurer, SchedulingConfigurer {
@Autowired
private AppConfigProperties appConfigProperties;
@Autowired
private Synchronizer synchronizer;
@Override
public Executor getAsyncExecutor() {
ThreadPoolTaskExecutor executor = new ThreadPoolTaskExecutor();
executor.setCorePoolSize(appConfigProperties.getThreadpoolCorePoolSize());
executor.setMaxPoolSize(appConfigProperties.getThreadpoolMaxPoolSize());
executor.setQueueCapacity(appConfigProperties.getThreadpoolQueueCapacity());
executor.setThreadNamePrefix("threadPoolExecutor-");
executor.initialize();
return executor;
}
@Override
public AsyncUncaughtExceptionHandler getAsyncUncaughtExceptionHandler() {
return new AsyncExceptionHandler();
}
@Override
public void configureTasks(ScheduledTaskRegistrar taskRegistrar) {
taskRegistrar.addCronTask(new Runnable() {
@Override
@Transactional
public void run() {
synchronizer.synchronizeAuto();
}
}, appConfigProperties.getCronExpression());
}
}
class MyService 调用 Spring JPA 存储库以获取所有 "Test" 个实例
一个 "Test" 实例有延迟加载 (MyLazyObject)
无论如何,如果我从我的控制器调用方法,一切都会像魅力一样。
当它来自调度程序 运行 时,我收到以下错误:
org.hibernate.LazyInitializationException: failed to lazily initialize a collection of role: com.tinea.apix.service.model.entity.apim.APIManagerEntity.synchroHistory, could not initialize proxy - no Session
有什么想法吗?
由于使用了在配置时调用的configureTasks
,所以很早就创建了Syncronizer
。太早了,它不再符合代理 creation/post 处理的条件。这反过来至少导致任务使用未代理的实例并且没有应用 @Transactional
。
相反,您应该使用 @Scheduled
注释和 cronString
属性 来像现在一样解决它。
@Scheduled(cron="@appConfigProperties.cronExpression")
SpEL 表达式中的 @
符号表示应解析具有给定名称的 bean。
@Scheduled(cron = "0 30 17 * * ?")
否则我们也可以这样使用