组件扫描接口上的自定义注释

时间:2013-07-04 19:44:40

标签: spring

我有一个组件扫描配置:

   @Configuration
   @ComponentScan(basePackageClasses = {ITest.class},
                  includeFilters = {@ComponentScan.Filter(type = FilterType.ANNOTATION, value = JdbiRepository.class)})
   public class MyConfig {

   }

基本上我想创建具有JdbiRepository注释

的扫描界面
@JdbiRepository
public interface ITest {
  Integer deleteUserSession(String id);
}

我想创建我的接口的代理实现。为此,我注册了一个自定义SmartInstantiationAwareBeanPostProcessor,它基本上创建了必要的实例,但上面的配置不会扫描具有JdbiRepository注释的接口。

如何通过自定义注释扫描界面?

编辑:

org.springframework.context.annotation.ClassPathScanningCandidateComponentProvider#isCandidateComponent似乎只接受具体的课程。

/**
 * Determine whether the given bean definition qualifies as candidate.
 * <p>The default implementation checks whether the class is concrete
 * (i.e. not abstract and not an interface). Can be overridden in subclasses.
 * @param beanDefinition the bean definition to check
 * @return whether the bean definition qualifies as a candidate component
 */
protected boolean isCandidateComponent(AnnotatedBeanDefinition beanDefinition) {
    return (beanDefinition.getMetadata().isConcrete() && beanDefinition.getMetadata().isIndependent());
}

编辑:

@Target({ElementType.TYPE})
@Retention(RetentionPolicy.RUNTIME)
@Documented
@Component
public @interface JdbiRepository {

   /**
    * The value may indicate a suggestion for a logical component name,
    * to be turned into a Spring bean in case of an autodetected component.
    * @return the suggested component name, if any
    */
   String value() default "";
}

3 个答案:

答案 0 :(得分:3)

正如我之前所说,组件扫描仅适用于具体类。

为了解决我的问题,我遵循了以下步骤:

  1. 实施了自定义org.springframework.context.annotation.ImportBeanDefinitionRegistrar
  2. 创建了导入我的自定义导入程序的自定义批注EnableJdbiRepositories
  3. 扩展ClassPathScanningCandidateComponentProvider以便扫描接口。我的自定义导入程序也使用此类来扫描接口。

答案 1 :(得分:0)

您可以创建一个虚拟实现并相应地对其进行注释,而不是扫描带注释的接口:

@JdbiRepository
public class DummyTest implements ITest {
  public Integer deleteUserSession(String id) {
    // nothing here, just being dummy
  }
}

然后,扫描虚拟实现basePackageClasses = {DummyTest.class}

这是一种解决方法,但非常简单,足以用于测试目的(似乎在这里)。

答案 2 :(得分:0)

创建一个虚拟实现对我来说似乎非常讨厌,所提到的所有步骤Cemo都需要付出很多努力。但扩展ClassPathScanningCandidateComponentProvider是最快的方法:

ClassPathScanningCandidateComponentProvider scanningProvider = new ClassPathScanningCandidateComponentProvider(false) {
    @Override
    protected boolean isCandidateComponent(AnnotatedBeanDefinition beanDefinition) {
        return true;
    }
};

现在,您还可以使用Spring扫描接口(自定义)注释 - 也适用于Spring Boot胖jar环境,其中fast-classpath-scanner(在this so q&a)可能会有一些限制。