Skip to content

Latest commit

 

History

History
262 lines (189 loc) · 8.1 KB

00011-spring5-annotation-based-container-configuration.md

File metadata and controls

262 lines (189 loc) · 8.1 KB

[toc]

Spring的容器配置可以有两种方式,一种基于XML文件, 一种基于注解。注解注入在XML注入之前执行。因此,当两个同时使用时,XML配置会覆盖注解注入的属性。

本文会主要介绍 @Required,@Autowired, @PostConstruct, @PreDestroy 和 @Resource 这几个注解。

这几个注解都是由 context:annotation-config/ 来引入的。本质上引入这个配置会隐式的注册AutoWiredAnnotationBeanPostProcessor(提供@Autowired),CommonAnnotationBeanPostProcessor(提供@PostConstruct, @PreDestroy, @Resource),RequiredAnnotationBeanPostProcessor(提供 @Required),从而提供各个注解的功能。

下面我们会分别介绍各个注解的功能。

@Required

@Required 一般用在方法上面,表示该方法的参数必须能通过配置或者自动装载来填充。通常如果某个属性是必须的,我们会使用这个注解。

不过从Spring Framework 5.1开始,@Required注解正式被弃用,取而代之的是使用构造函数注入用于所需的属性,或使用InitializingBean.afterPropertiesSet()的自定义实现以及bean属性setter方法。

示例代码如下:

public class RequiredBean {

    private BeanA  beanA;

    @Required
    public void setBeanA(BeanA beanA){
        this.beanA=beanA;

    }
}

@Autowired

@Autowired 就是自动注入所需要的字段,参数等。JSR 330的@Inject注解可以代替spring的@Autowired注解。

你可以将@Autowired注解到构造器中,如下所示:

public class AutowiredBean {

    private BeanA beanA;

    @Autowired
    public AutowiredBean(BeanA beanA){
        this.beanA=beanA;
    }
}

从SpringFramework4.3开始,如果目标bean只定义了一个构造函数,那么就不再需要在此类构造函数上使用@Autowired注解。但是,如果有多个构造函数可用,则必须至少对其中一个进行注解,以告诉容器使用哪一个。

@Autowired也可以注解到传统的setter方法,如下例子所示:

public class AutowiredBean {

    private BeanB beanB;

    @Autowired
    public void setBeanB(BeanB beanB){
        this.beanB=beanB;
    }
}

也可以把注解应用到任何名字和多个参数,如下所示:

    @Autowired
    public void configAB(BeanA beanA , BeanB beanB){
        this.beanA=beanA;
        this.beanB=beanB;
        
    }

@Autowired也可以用在字段上,如下所示:

    @Autowired
    private BeanC beanC;

还可以通过将注解添加到需要该类型数组的字段或方法,那么可以从ApplicationContext中获取到该特定类型的所有bean,如下例所示:

    @Autowired
    private BeanC[] beanCList;

如果希望数组或列表中的项按特定顺序排序,目标bean可以实现org.springframework.core.Ordered接口,或者可以使用@Order或标准的@Priority注解。 否则,它们的顺序遵循容器中相应目标bean定义的注册顺序。

Map实例也可以被注入,只要key是String类型。Map value包括了所有的类型匹配的Bean,keys是该bean的名字。如下所示:

    @Autowired
    public void configMapA(Map<String,BeanA> mapA){
    this.mapA=mapA;
    }

@Autowired有个required属性,如果要注入的bean有可能不存在,则可以如下所示:

    @Autowired(required = false)
    public void setBeanC(BeanC beanC){
    }

建议使用@Autowired的'required'属性而不是使用setter方法上的@Required注解。“required”属性表示自动装载需要该属性, 如果无法自动装载,则忽略该属性。而对于@Required来说,如果未定义任何值,则会报异常。

也可以通过Java 8的java.util.Optional表示特定依赖项的非必需性质,如下示例显示:

    @Autowired
    public void setMovieFinder(Optional<BeanC> BeanC) {
    }

在Spring Framework 5.0中,你也可以使用@Nullable注解:

    @Autowired
    public void setMovieFinderC(@Nullable BeanC beanC) {
    }

Spring可以使用@Autowired来自动解析一些默认存在的bean如:BeanFactory、ApplicationContext、Environment、ResourceLoader、ApplicationEventPublisher和MessageSource。这些接口及其扩展接口(如ConfigurableApplicationContext或ResourcePatternResolver)。 如下所示,自动注入ApplicationContext:

    @Autowired
    private ApplicationContext context;

注意: @Autowired, @Inject, @Value, 和 @Resource 注解是在Spring的BeanPostProcessor中处理的,这意味着你不能将这些注解用在你自己的BeanPostProcessor,BeanFactoryPostProcessor类型。

@primary

当按类型注入的时候,可能会有多个候选项,则可以通过@Primary注解表示优先的对象。如下所示:

@Configuration
public class ConfigBean {

    @Bean
    @Primary
    public BeanA firstBeanA() { return new BeanA(); }

    @Bean
    public BeanA secondBeanA() {  return new BeanA();}

}

@Qualifier

@Primary是一种在多个实例中按类型使用自动装载的有效方法,但是如果你希望对注入的Bean进行更细粒度的控制时候,可以使用@Qualifier。如下所示:

    @Bean
    @Qualifier("main")
    public BeanC beanC() {  return new BeanC();}
    @Autowired
    @Qualifier("main")
    private BeanA beanA;

    @Autowired
    public void setBeanA(@Qualifier("main") BeanA beanA){

    }

限定符的值并不是唯一的,它只是一个过滤标准。

@Autowired一般用来通过类型匹配,@Resource则是通过名称匹配。

也可以创建自定义注解:

@Target({ElementType.FIELD, ElementType.PARAMETER})
@Retention(RetentionPolicy.RUNTIME)
@Qualifier
public @interface Genre {

        String value();
}

泛型

除了@Qualifier注解外,还可以使用Java泛型类型作为隐式的限定形式。例如,假设您具有以下配置:

public class StringStore implements Store<String> {
}

public class IntegerStore implements Store<Integer> {
}
    @Bean
    public StringStore stringStore() {
        return new StringStore();
    }

    @Bean
    public IntegerStore integerStore() {
        return new IntegerStore();
    }
public class GenericBean {

    @Autowired
    private Store<String> s1; // <String> qualifier, injects the stringStore bean

    @Autowired
    private Store<Integer> s2; // <Integer> qualifier, injects the integerStore bean

    // Inject all Store beans as long as they have an <Integer> generic
    // Store<String> beans will not appear in this list
    @Autowired
    private List<Store<Integer>> s;

}

@Resource

@Resource用在字段或者Setter方法上,默认情况下@Resource通过名字来注入。

public class ResourceBean {
    
    @Resource(name = "beanA")
    private BeanA BeanA;
}

如果未显式指定名称,则从字段名或setter方法派生默认名称.

在@Resource用法中,如果没有指定显式名称,并且类似于@Autowired,@Resource会找到一个主类型匹配,而不是指定的bean,并解析已知的可解析依赖项:BeanFactory、ApplicationContext、ResourceLoader、ApplicationEventPublisher,和MessageSource接口。

@PostConstruct和@PreDestroy

这两个注解主要用做生命周期回调。如下所示:

public class ConstructBean   {

    @PostConstruct
    public void populateMovieCache() {
        // populates the movie cache upon initialization...
    }

    @PreDestroy
    public void clearMovieCache() {
        // clears the movie cache upon destruction...
    }
}

与@Resource一样,@PostConstruct和@PreDestroy注解类型是JDK 6到8标准Java库的一部分。然而,整个javax.annotation包与JDK 9中的核心Java模块分离,并最终在JDK 11中被删除。如果需要,javax.annotation-api工件现在需要通过maven central获得,只需像其他库一样添加到应用程序的类路径中即可。

本文的代码可以参考annotation-config

更多教程请参考 flydean的博客