Определение bean-компонента CacheManager в Config.class приводит к NoSuchBeanDefinitionException

У меня есть служба Spring, которая проверяет записи базы данных. Чтобы свести к минимуму вызовы моего репозитория, оба метода поиска имеют значение «@Cacheable». Но когда я пытаюсь запустить свой сервисный компонент, в то время как мой класс конфигурации имеет определение компонента CacheManager, я получаю следующее исключение NoSuchBeanDefinitionException:

Caused by: org.springframework.beans.factory.NoSuchBeanDefinitionException: No qualifying bean of type 'foo.mediacode.directory.MediaCodeDirectoryService' available
at org.springframework.beans.factory.support.DefaultListableBeanFactory.getBean(DefaultListableBeanFactory.java:353)
at org.springframework.beans.factory.support.DefaultListableBeanFactory.getBean(DefaultListableBeanFactory.java:340)
at org.springframework.context.support.AbstractApplicationContext.getBean(AbstractApplicationContext.java:1093)
at foo.mediacode.directory.MediaCodeDirectoryService.implementation(MediaCodeDirectoryService.java:63)
at foo.campaigntree.directory.CampaignTreeDirectoryService.<init>(CampaignTreeDirectoryService.java:18)
... 15 more

Если я уберу определение компонента CacheManager, я смогу инициализировать свой сервисный компонент, и он будет работать без каких-либо проблем и кэширования!

Вот мой код: Конфигурация

...
    @Configuration
    @EnableCaching
    @EnableJpaRepositories(...)
    @PropertySource({...})
    public class MediaCodeDirectoryServiceConfig {

        private static Logger   configLogger    = Logger.getLogger(MediaCodeDirectoryServiceConfig.class.getName());

        @Value("${jpa.loggingLevel:FINE}")
        private String          loggingLevel;

        @Value("${mysql.databaseDriver}")
        private String          dataBaseDriver;

        @Value("${mysql.username}")
        private String          username;

        @Value("${mysql.password}")
        private String          password;

        @Value("${mysql.databaseUrl}")
        private String          databaseUrl;

        @Bean
        public static PropertySourcesPlaceholderConfigurer propertyConfigInDev() {
            ...
        }

        @Bean
        public MediaCodeDirectoryService mediaCodeDirectoryService() {
            return new MediaCodeDirectoryService();
        }

        @Bean
        public CacheManager mediaCodeCacheManager() {
            SimpleCacheManager cacheManager = new SimpleCacheManager();
            cacheManager.setCaches(Arrays.asList(new ConcurrentMapCache("mediaCodeMappingRegexCache"),
                    new ConcurrentMapCache("mediaCodeMappingsCache")));

            return cacheManager;
        }

        @Bean
        public JpaTransactionManager transactionManager() {
            ...
        }

        @Bean
        public LocalContainerEntityManagerFactoryBean entityManagerFactory() {
            ...
        }

        public DataSource getDataSource() {
            ...
        }

        public JpaDialect getJpaDialect() {
            ...
        }

        public Properties getEclipseLinkProperty() {
            ...
        }

        public JpaVendorAdapter getJpaVendorAdapter() {
            ...
        }
    }

Услуга

....
    public class MediaCodeDirectoryService implements MediaCodeDirectoryServiceApi {

        ...

        @Autowired
        private MediaCodeDirectoryRepository repo;

        @SuppressWarnings("resource")
        public static MediaCodeDirectoryServiceApi implementation() {
        if (INSTANCE == null) {
                ApplicationContext ctx = new AnnotationConfigApplicationContext(MediaCodeDirectoryServiceConfig.class);
                INSTANCE = ctx.getBean(MediaCodeDirectoryService.class);
            }

            return INSTANCE;
        }
...

Репозиторий

...
@Repository
public interface MediaCodeDirectoryRepository extends CrudRepository<MediaCodeDao, Integer> {

    @Cacheable("mediaCodeMappingRegexes")
    @Query("SELECT m FROM  #{#entityName} m WHERE (m.fooId = :fooId) AND (m.isRegex = :isRegex) ORDER BY (m.orderId DESC, m.id ASC)")
    List<MediaCodeDao> findByfooIdAndIsRegexOrderByOrderIdDescAndIdAsc(@Param("fooId") int fooId, @Param("isRegex") boolean isRegex);

    @Cacheable("mediaCodeMappings")
    List<MediaCodeDao> findByMediaCode(String MediaCode, Pageable pageable);
}

Когда я отлаживаю в DefaultListableBeanFactory, я могу найти в beanDefinitionMap мой mediaCodeDirectoryService, а также в beanDefinitionNames появляется mediaCodeDirectoryService. Но DefaultListableBeanFactory.getBean(...) не может разрешить имя, а namedBean в строке 364 имеет значение null.

Когда я пытаюсь получить контекст через String, например:

INSTANCE = (MediaCodeDirectoryService) ctx.getBean("mediaCodeDirecotryService")

Я избегаю исключения NoSuchBeanDefinitionException, но сталкиваюсь с другим.

Кто-нибудь здесь имеет представление о том, что может быть причиной этого? Я что-то пропустил в своей конфигурации? Спасибо!


person grimbo    schedule 11.05.2017    source источник
comment
Вы должны программировать для взаимодействия НЕ с конкретными классами... Поэтому вместо MediaCodeDirectoryService используйте MediaCodeDirectoryServiceApi в своем коде.   -  person M. Deinum    schedule 12.05.2017
comment
на другом узле ваш код опасен, вы НИКОГДА не должны вызывать new *ApplicationContext, поскольку вы в основном воссоздаете свое приложение снова и снова. Если у вас есть этот код в нескольких местах, вы в конечном итоге получите проблемы с памятью, проблемы с транзакциями, странные проблемы с параллелизмом (конечно, если вы хотите, чтобы они не стеснялись делать это, но я настоятельно рекомендую вам не делать этого!).   -  person M. Deinum    schedule 12.05.2017


Ответы (1)


Кэширование применяется через АОП. Для AOP Spring использует подход на основе прокси, и по умолчанию создаются прокси на основе интерфейса.

public class MediaCodeDirectoryService implements MediaCodeDirectoryServiceApi {... }

С этим определением класса во время выполнения вы получите динамически созданный класс (Proxy$51 или что-то в этом роде), который реализует все интерфейсы, но это не MediaCodeDirectoryService. Однако это MediaCodeDirectoryServiceApi.

У вас есть 2 способа исправить это: либо программа для интерфейсов (что вы должны были делать в любом случае, потому что вы определили интерфейсы) вместо конкретных классов, либо использовать прокси на основе классов.

Первый вариант включает в себя изменение вашего кода в местах непосредственно @Autowire или получение экземпляра MediaCodeDirectoryService для использования вместо него MediaCodeDirectoryServiceApi (что, по моему мнению, вы уже должны были сделать, зачем еще определять интерфейс). Теперь вы получите прокси, и все будет работать.

Второй вариант предполагает установку proxyTargetClass=true в аннотации @EnableCaching. Тогда вместо прокси на основе интерфейса вы получите прокси на основе класса.

@EnableCaching(proxyTargetClass=true)
person M. Deinum    schedule 12.05.2017
comment
Большое спасибо за ваш четкий и подробный ответ. После этого я провел некоторое исследование и нашел это общее объяснение того, почему для автоматического подключения интерфейса вместо класса stackoverflow.com/questions/12899372/ - person grimbo; 12.05.2017