带条件注释的原型范围bean

时间:2016-11-27 07:53:00

标签: java spring

我尝试在每个具有可能不同的实现的请求上重新加载bean。

在我的控制器中,我每次从ApplicationContext检索bean:

@Controller
public class LabelsController implements ApplicationContextAware {

    private ApplicationContext applicationContext;

    @RequestMapping("/...")
    public ModelAndView labelConcerns()  {
        System.out.println("Here I ask for a fresh bean");
        InconsistentCaseDetector inconsistentCaseDetector = applicationContext.getBean(InconsistentCaseDetector.class);

InconsistentCaseDetector是一个原始界面。我有几个实现注释如下(具有不同的条件):

@Component
@Conditional(SkosFormatSelected.class)
@Scope(value = "prototype", proxyMode = ScopedProxyMode.INTERFACES)
public class InconsistentCaseDetectorImpl implements InconsistentCaseDetector {
...

条件的一个例子:

public class SkosFormatSelected implements Condition {

    @Override
    public boolean matches(ConditionContext context, AnnotatedTypeMetadata metadata) {
        System.out.println("Is skos format selected ?");
        return Formats.getCurrent().equals(Formats.SKOS);
    }

但条件类的matches方法仅在启动时调用。 在启动过程中,会显示日志说明(“是否选择了skos格式?”,“是否选择了xml格式?”......),然后显示:

org.springframework.context.annotation.ClassPathBeanDefinitionScanner  - Identified candidate component class: file [/home/..../.../WEB-INF/classes/tests/model/vocabulary/skos/algorithm/InconsistentCaseDetectorImpl.class] 

但是,然后,在每个请求中,条件不再执行,并且它始终与所服务的实现相同。 这是日志在运行时显示的内容:

Here I ask for a fresh bean
163326 [http-nio-8080-exec-23] DEBUG org.springframework.beans.factory.support.DefaultListableBeanFactory  - Returning cached instance of singleton bean 'inconsistentCaseDetectorImpl'
...
163328 [http-nio-8080-exec-23] DEBUG org.springframework.beans.factory.support.DefaultListableBeanFactory  - Creating instance of bean 'scopedTarget.inconsistentCaseDetectorImpl'
163329 [http-nio-8080-exec-23] DEBUG org.springframework.beans.factory.support.DefaultListableBeanFactory  - Finished creating instance of bean 'scopedTarget.inconsistentCaseDetectorImpl'

因此,当我请求bean时,它从缓存中返回一个实例而不是创建一个新实例。后来,但我不知道在哪里,它似乎创建了一个新的,仍然没有执行条件子句。

1 个答案:

答案 0 :(得分:0)

使用

@Autowired
@RequestMapping("/...")
public ModelAndView labelConcerns(final InconsistentCaseDetector inconsistentCaseDetector)  {
    ...
}

应该有用。

顺便说一下,

  1. 您应该将@Component移动到界面(DRY / SRP),也可以移动@Scope
  2. 您应该注入ApplicationContext而不是使用界面ApplicationContextAware(DIP)