Config.class 中的 CacheManager bean 定义导致 NoSuchBeanDefinitionException

gri*_*mbo 3 java eclipselink spring-data-jpa spring-cache

我有一个正在检查数据库条目的 Spring 服务。为了最小化我的存储库调用,两个查找方法都是“@Cacheable”。但是当我尝试初始化我的服务 bean 而我的配置类有一个 CacheManager bean 定义时,我得到以下 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
Run Code Online (Sandbox Code Playgroud)

如果我取出 CacheManager bean 定义,我可以初始化我的服务 bean 并且它运行没有任何问题和缓存

这是我的代码:配置

...
    @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() {
            ...
        }
    }
Run Code Online (Sandbox Code Playgroud)

服务

....
    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;
        }
...
Run Code Online (Sandbox Code Playgroud)

存储库

...
@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);
}
Run Code Online (Sandbox Code Playgroud)

当我调试到DefaultListableBeanFactory 时,我可以在 beanDefinitionMap 中找到我的mediaCodeDirectoryService并且在 beanDefinitionNames mediaCodeDirectoryService 中出现。但是 DefaultListableBeanFactory.getBean(...) 无法解析 name 并且第 364 行中的namedBean为空。

当我尝试通过 String 获取上下文时,例如:

INSTANCE = (MediaCodeDirectoryService) ctx.getBean("mediaCodeDirecotryService")
Run Code Online (Sandbox Code Playgroud)

我避免了 NoSuchBeanDefinitionException 但我遇到了另一个。

这里有人知道这可能是什么原因吗?我在配置中遗漏了什么吗?谢谢!

M. *_*num 5

缓存是通过 AOP 应用的。对于 AOP Spring 使用基于代理的方法,默认是创建基于接口的代理。

public class MediaCodeDirectoryService implements MediaCodeDirectoryServiceApi {... }
Run Code Online (Sandbox Code Playgroud)

在运行时使用此类定义,您将获得一个动态创建的类(Proxy$51或类似的东西),它实现了所有接口,但它不是MediaCodeDirectoryService. 然而,它是一个MediaCodeDirectoryServiceApi.

您有两种方法可以解决这个问题,要么编程到接口(因为您已经定义了接口,所以无论如何您都应该这样做)而不是具体类或使用基于类的代理。

第一个选项涉及您直接在地方更改代码@Autowire或获取MediaCodeDirectoryService要使用的实例MediaCodeDirectoryServiceApi(恕我直言,您应该已经这样做了,为什么还要定义接口)。现在您将注入代理,一切都会正常进行。

第二个选项涉及您设置proxyTargetClass=true你的@EnableCaching注释。然后,您将获得一个基于类的代理,而不是基于接口的代理。

@EnableCaching(proxyTargetClass=true)
Run Code Online (Sandbox Code Playgroud)