@Scheduled + Hibernate -> LazyInitializationException - PullRequest
0 голосов
/ 29 октября 2018

Я нахожусь под Spring Boot 2.0.5, используя Spring Data JPA

У меня есть такой класс (для понимания):

@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());
    }
}

Класс MyService вызывает репозиторий Spring JPA для получения всех «тестовых» экземпляров

Экземпляр "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

Есть идеи?

1 Ответ

0 голосов
/ 29 октября 2018

Из-за использования configureTasks, который вызывается во время конфигурации, Syncronizer создается очень рано. Настолько рано, что он больше не подходит для создания / пост-обработки прокси. Что, в свою очередь, приводит, по крайней мере к задаче, к использованию незащищенного экземпляра без применения @Transactional.

Вместо этого вы должны использовать аннотацию @Scheduled вместе со свойством cronString, чтобы разрешить ее так же, как вы делаете сейчас.

@Scheduled(cron="@appConfigProperties.cronExpression")

Символ @ в выражении SpEL указывает, что компонент с указанным именем должен быть разрешен.

Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...