`

spring beans常用的注解 | 注解bean的定义AnnotatedBeanDefinition | 注解bean的实现类 | 注解的解析与处理

 
阅读更多

 

原文  http://www.cnblogs.com/davidwang456/p/4199459.html

 http://www.tuicool.com/articles/rIVr2a

 

随着spring注解的引入,越来越多的开发者开始使用注解,这篇文章将对注解的机制进行串联式的讲解,不求深入透彻,但求串起spring beans注解的珍珠,展示给大家。

1. spring beans常用的注解:

public @interface Autowired:可以对成员变量、方法和构造函数进行标注,来完成自动装配的工作。

Marks a constructor, field, setter method or config method as to be autowired by Spring's dependency injection facilities.
Only one constructor (at max) of any given bean class may carry this annotation, indicating the constructor to autowire when used as a Spring bean. Such a constructor does not have to be public.

Fields are injected right after construction of a bean, before any config methods are invoked. Such a config field does not have to be public.

Config methods may have an arbitrary name and any number of arguments; each of those arguments will be autowired with a matching bean in the Spring container. Bean property setter methods are effectively just a special case of such a general config method. Such config methods do not have to be public.

In the case of multiple argument methods, the 'required' parameter is applicable for all arguments.

In case of a Collection or Map dependency type, the container will autowire all beans matching the declared value type. In case of a Map, the keys must be declared as type String and will be resolved to the corresponding bean names.

Note that actual injection is performed through a BeanPostProcessor which in turn means that you cannot use @Autowired to inject references into BeanPostProcessor or BeanFactoryPostProcessor types. Please consult the javadoc for the AutowiredAnnotationBeanPostProcessor class (which, by default, checks for the presence of this annotation).

Since:
2.5

public @interface Configurable @Configurable 注解中的autowire属性就可以让Spring来自动装配了:  @Configurable(autowire=Autowire.BY_TYPE) 或者  @Configurable(autowire=Autowire.BY_NAME ,这样就可以按类型或者按名字自动装配了。

Marks a class as being eligible for Spring-driven configuration.
Typically used with the AspectJ AnnotationBeanConfigurerAspect.

Since:
2.0
public @interface Value:用于注入SpEL表达式,可以放置在字段方法或参数上。
Annotation at the field or method/constructor parameter level that indicates a default value expression for the affected argument.
Typically used for expression-driven dependency injection. Also supported for dynamic resolution of handler method parameters, e.g. in Spring MVC.

A common use case is to assign default field values using "#{systemProperties.myProp}" style expressions.

Note that actual processing of the @Value annotation is performed by a BeanPostProcessor which in turn means that you cannot use @Value within BeanPostProcessor or BeanFactoryPostProcessor types. Please consult the javadoc for the AutowiredAnnotationBeanPostProcessor class (which, by default, checks for the presence of this annotation).

Since:
3.0
public @interface Qualifier:指定限定描述符,对应于基于XML配置中的<qualifier>标签,@Qualifier限定描述符除了能根据名字进行注入,但能进行更细粒度的控制如何选择候选者
@Qualifier(value = "限定标识符") 。
This annotation may be used on a field or parameter as a qualifier for candidate beans when autowiring. It may also be used to annotate other custom annotations that can then in turn be used as qualifiers.
Since:
2.5
public @interface Required 依赖检查;
Marks a method (typically a JavaBean setter method) as being 'required': that is, the setter method must be configured to be dependency-injected with a value.
Please do consult the javadoc for the RequiredAnnotationBeanPostProcessor class (which, by default, checks for the presence of this annotation).

Since:
2.0

2. 注解bean的定义AnnotatedBeanDefinition

public interface AnnotatedBeanDefinition extends BeanDefinition {/**
     * Obtain the annotation metadata (as well as basic class metadata)
     * for this bean definition's bean class.
     * @return the annotation metadata object (never {@code null})
     */
    AnnotationMetadata getMetadata();

}

该接口继承了BeanDefinition,提供了一个getMetadata()方法来获取该bean definition的注解元数据。

其中,AnnotationMetadata定义了访问特定类的注解的抽象接口,它不需要加载该类即可访问。

public interface AnnotationMetadata extends ClassMetadata, AnnotatedTypeMetadata {
}
ClassMetadata定义了一个特定类的抽象元数据,不需要加载此类。主要方法如下:
String getClassName()返回该类的名称。boolean isInterface()返回该类是否是接口。boolean isAbstract()返回该类是否为抽象类。boolean isConcrete()返回该类是否为具体类。boolean isFinal()返回该类是否为finalboolean hasSuperClass()返回该类是否有父类
String getSuperClassName()返回父类的名称,没有的话返回null.
String[] getInterfaceNames()返回继承的接口数组,如果没有,返回空.
String[] getMemberClassNames()返回引用的类的名称。

AnnotatedTypeMetadata定义访问特定类型的注解,不需要加载类。主要方法有:

boolean isAnnotated(String annotationType)是否有匹配的注解类型
Map<String,Object> getAnnotationAttributes(String annotationType,boolean classValuesAsString)获取特定类型注解的属性

AnnotationMetadata的标准实现类StandardAnnotationMetadata,它使用标准的反射来获取制定类的内部注解信息。主要方法有:

getAllAnnotationAttributes(String annotationType) 
getAnnotatedMethods(String annotationType) 
hasMetaAnnotation(String annotationType) 
isAnnotated(String annotationType) 
hasAnnotatedMethods(String annotationType) 

AnnotationMetadata还有一个子类:AnnotationMetadataReadingVisitor,它是字节码访问实现。

class ClassMetadataReadingVisitor extends ClassVisitor implements ClassMetadata {
}

让我们了解一下visitor模式:

定义:

The Gang of Four defines the Visitor as:  " Represent an operation to be performed on elements of an object structure. Visitor lets you define a new operation without changing the classes of the elements on which it operates. " —″≤

The nature of the Visitor makes it an ideal pattern to plug into public APIs thus allowing its clients to perform operations on a class using a “visiting” class without having to modify the source.

uml 结构图如下:

小结:vistor设置模式把状态抽象出来成为一个接口(访问者),不同的状态就作为状态的不同实现类(不同的访问者)。

3. 注解bean的实现类

3.1 AnnotatedGenericBeanDefinition

继承了GenericBeanDefinition,增加了对注解元素的支持,这种支持是通过AnnotationBeanDefinition暴露的的注解元素接口。

GenericBeanDefinition主要用来测试AnnotatedBeanDefinition上的操作的,例如:在spring的component扫描支持的实现中(默认实现类是ScannedGenericBeanDefinition,它同样实现了AnnotatedBeanDefinition接口)

3.2 ConfigurationClassBeanDefinition

ConfigurationClassBeanDefinition是ConfigurationClassBeanDefinitionReader的内部类,ConfigurationClassBeanDefinitionReader 读取一组完全填充了属性的配置实例,通过context内给定的BeanDefinitionRegistry进行注册bean definition。这个类在BeanDefinitionReader这层后就改造,但没有继承或者扩展配置类。

3.3 ScannedGenericBeanDefinition

基于asm的类解析器,是GenericBeanDefinition类的扩展,支持注解元数据,这种支持通过AnnotatedBeanDefinition接口实现。

4. 注解的解析与处理

4.1 @Autowired注解实现AutowiredAnnotationBeanPostProcessor

public class AutowiredAnnotationBeanPostProcessor extends InstantiationAwareBeanPostProcessorAdapter
        implements MergedBeanDefinitionPostProcessor, PriorityOrdered, BeanFactoryAware {
}
AutowiredAnnotationBeanPostProcessor 间接继承了BeanPostProcessor,它自动绑定注解的field,setter方法和任意的配置方法。当检测到5个java注解时这些成员被注入其中。spring默认的注解为@Autowired@Value。
另外:也支持JSR-330@inject注解,作为@Autowired的替代方案。

当制定bean 类的唯一构造方法带有required 注解参数,且required值设置为true时,表明当作为spring一个bean时,构造方法默认自动绑定。若多个构造方法带有non-required 注解参数,它们将作为自动绑定的候选项。带有大量依赖的构造方法可以通过spring容器中的匹配的bean来构造,如果没有候选者满足条件,则会使用默认的构造器。注解构造器不一定是public的。
Field注入是在构造方法之后,配置方法之前,这种配置field不要求一定为public
配置方法可以有任意的名称和不定的参数列表,这些参数则被自动注入到spring容器中的匹配的bean。bean的属性setter方法仅仅是通用的配置方法的一个特例而已。配置方法不要求方法一定为public
注意:默认注册AutowiredAnnotationBeanPostProcessor的方式有<context:annotation-config> 和<context:component-scan> xml标签,如果你指定了一个自定义的AutowiredAnnotationBeanPostProcessor bean definition,移除或者关闭默认的注解配置。
其中,MergedBeanDefinitionPostProcessor的定义如下:
public interface MergedBeanDefinitionPostProcessor extends BeanPostProcessor {/**
     * Post-process the given merged bean definition for the specified bean.
     * @param beanDefinition the merged bean definition for the bean
     * @param beanType the actual type of the managed bean instance
     * @param beanName the name of the bean
     */
    void postProcessMergedBeanDefinition(RootBeanDefinition beanDefinition, Class<?> beanType, String beanName);

}
BeanPostProcessor 是一个可以定制修改一个新的bean实例的工厂钩子,例如:检查marker接口或者使用代理包装他们。
applicationContext可以在他们的bean容器中自动识别BeanPostProcessor bean,并将它们应用到接下来所创建的bean。一般的bean factory通过编程来注册Post-processor,并将它们应用到整个bean factory创建bean的过程中。

通常意义上,post-processor 设置bean属性通过marker 接口或者类似于实现postProcessBeforeInitialization(java.lang.Object, java.lang.String);使用代理包装bean通常实现postProcessAfterInitialization(java.lang.Object, java.lang.String).

4.2 @configurable注解实现AnnotationBeanWiringInfoResolver

设置 @Configurable 注解中的autowire属性就可以让Spring来自动装配了:  @Configurable(autowire=Autowire.BY_TYPE) 或者  @Configurable(autowire=Autowire.BY_NAME ,这样就可以按类型或者按名字自动装配了。

AnnotationBeanWiringInfoResolver 继承自BeanWiringInfoResolver,BeanWiringInfoResolver使用configurable注解来查找哪些类需要自动绑定。

public class AnnotationBeanWiringInfoResolver implements BeanWiringInfoResolver {
}

实现了BeanWiringInfoResolver的resolveWiringInfo方法

@Override
public BeanWiringInfo resolveWiringInfo(Object beanInstance) {
Assert.notNull(beanInstance, "Bean instance must not be null");
Configurable annotation = beanInstance.getClass().getAnnotation(Configurable.class);
return (annotation != null ? buildWiringInfo(beanInstance, annotation) : null);
}
/**
 * Build the BeanWiringInfo for the given Configurable annotation.
 * @param beanInstance the bean instance
 * @param annotation the Configurable annotation found on the bean class
 * @return the resolved BeanWiringInfo
 */
protected BeanWiringInfo buildWiringInfo(Object beanInstance, Configurable annotation) {
if (!Autowire.NO.equals(annotation.autowire())) {
return new BeanWiringInfo(annotation.autowire().value(), annotation.dependencyCheck());
}
else {
if (!"".equals(annotation.value())) {
// explicitly specified bean name
return new BeanWiringInfo(annotation.value(), false);
}
else {
// default bean name
return new BeanWiringInfo(getDefaultBeanName(beanInstance), true);
}
}
}

4.3 @qualifier的注解实现类QualifierAnnotationAutowireCandidateResolver

public class QualifierAnnotationAutowireCandidateResolver extends GenericTypeAwareAutowireCandidateResolver {
}

public class GenericTypeAwareAutowireCandidateResolver implements AutowireCandidateResolver, BeanFactoryAware {
}

其中,AutowireCandidateResolver是一个策略接口,由它来决定特定的bean definition对特定的依赖是否可以作为一个自动绑定的候选项,它的主要方法有:

boolean isAutowireCandidate(BeanDefinitionHolder bdHolder, DependencyDescriptor descriptor)

Object getLazyResolutionProxyIfNecessary(DependencyDescriptor descriptor,String beanName)

Object getSuggestedValue(DependencyDescriptor descriptor)

QualifierAnnotationAutowireCandidateResolver间接实现了AutowireCandidateResolver,对要自动绑定的field或者参数和bean definition根据@qualifier注解进行匹配。同时也支持通过@value注解来绑定表达式的值。

另外,还只是JSR-330的javax.inject.Qualifier注解。

4.4 @required注解实现类RequiredAnnotationBeanPostProcessor。

public class RequiredAnnotationBeanPostProcessor extends InstantiationAwareBeanPostProcessorAdapter
        implements MergedBeanDefinitionPostProcessor, PriorityOrdered, BeanFactoryAware {
}

和AutowiredAnnotationBeanPostProcessor一样,间接继承自BeanPostProcessor,它增加了对javaBean属性配置的约束,java 5 注解可以检测bean的required属性,spring默认是@Required注解。

注意:默认注册AutowiredAnnotationBeanPostProcessor的方式有<context:annotation-config><context:component-scan> xml标签,如果你指定了一个自定义的
默认注册AutowiredAnnotationBeanPostProcessor的方式有<context:annotation-config><context:component-scan> xml标签,如果你指定了一个自定义的AutowiredAnnotationBeanPostProcessor bean definition,移除或者关闭默认的注解配置。其余和AutowiredAnnotationBeanPostProcessor类似,不一一赘述了。

4.5 初始化和销毁方法的注解实现类InitDestroyAnnotationBeanPostProcessor

public class InitDestroyAnnotationBeanPostProcessor
        implements DestructionAwareBeanPostProcessor, MergedBeanDefinitionPostProcessor, PriorityOrdered, Serializable {
}

InitDestroyAnnotationBeanPostProcessor间接继承了BeanPostProcess,实现了通过注解来初始化和销毁方法,是spring的InitializingBean和DisposableBean回调接口的注解实现。

它通过"initAnnotationType"和"destroyAnnotationType"属性来检查指定的注解类型,任何自定义的注解都可以使用。

初始化和销毁注解可以用在任意可见的方法:public,package-protected,protected,private等。尽管可以对多个方法进行注解,但我们推荐只在一个初始化和销毁方法上各自进行注解。

小结:

Spring3的基于注解实现Bean依赖注入支持如下三种注解:

Spring自带依赖注入注解: Spring自带的一套依赖注入注解;

JSR-250注解:Java平台的公共注解,是Java EE 5规范之一,在JDK6中默认包含这些注解,从Spring2.5开始支持。

JSR-330注解:Java 依赖注入标准,Java EE 6规范之一,可能在加入到未来JDK版本,从Spring3开始支持;

其中,

Spring自带依赖注入注解

1 @Required:依赖检查;

2 @Autowired:自动装配2 

自动装配,用于替代基于XML配置的自动装配

基于@Autowired的自动装配,默认是根据类型注入,可以用于构造器、字段、方法注入

3 @Value:注入SpEL表达式

用于注入SpEL表达式,可以放置在字段方法或参数上

@Value(value = "SpEL表达式")  

@Value(value = "#{message}")  

4 @Qualifier:限定描述符,用于细粒度选择候选者

@Qualifier限定描述符除了能根据名字进行注入,但能进行更细粒度的控制如何选择候选者

@Qualifier(value = "限定标识符") 

JSR-250注解

1 @Resource:自动装配,默认根据类型装配,如果指定name属性将根据名字装配,可以使用如下方式来指定

@Resource(name = "标识符")  

字段或setter方法 

2 @PostConstruct和PreDestroy:通过注解指定初始化和销毁方法定义

JSR-330注解

1 @Inject:等价于默认的@Autowired,只是没有required属性

2 @Named:指定Bean名字,对应于Spring自带@Qualifier中的缺省的根据Bean名字注入情况

3 @Qualifier:只对应于Spring自带@Qualifier中的扩展@Qualifier限定描述符注解,即只能扩展使用,没有value属性

参考文献:
   1. http://docs.spring.io/spring/docs/current/javadoc-api/org/springframework/beans/factory/annotation/
   2. http://blog.csdn.net/wangshfa/article/details/9712379



分享到:
评论

相关推荐

    自定义注解得使用,模拟spring通过注解方式创建bean实例

    现在,Spring会识别并处理带有`MyComponent`注解的类,根据注解的`value`属性来创建bean实例。这种方式减少了XML配置,使代码更加简洁、易读。 总结一下,自定义注解和基于注解的bean创建是Spring框架中强大的特性...

    spring3零配置注解实现Bean定义(包括JSR-250、JSR-330)

    ### Spring3零配置注解实现Bean定义 #### 概述 在Spring框架中,传统的Bean定义方式主要依赖于XML配置文件。随着技术的发展与需求的变化,Spring为了简化配置过程,引入了注解的方式进行Bean的定义。这不仅减少了...

    Spring 自定义注解的解析

    这可以通过定义一个使用`@Retention`和`@Target`注解的注解类来完成。`@Retention`决定了注解的生命周期,通常设置为`RetentionPolicy.RUNTIME`,以便在运行时可被程序访问。`@Target`指定注解可以应用到哪些元素...

    Spring Boot 常用注解.rar

    **Spring Boot 常用注解详解** Spring Boot以其简洁的配置、快速的启动和集成众多优秀框架的能力,成为Java开发领域中的热门选择。在Spring Boot应用中,注解起着至关重要的作用,它们简化了配置,使得代码更加简洁...

    Spring demo 自动检测注解

    6. **自定义依赖解析**:开发者可以通过实现`org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor`接口来自定义依赖解析逻辑。 7. **组件扫描**:Spring的`@Component`、`@Service...

    spring-beans.zip

    到了`4.2.0.RELEASE`,Spring Beans进一步优化了配置元数据处理,提升了性能,并添加了对`@Profile`注解的支持,使得bean可以根据特定的环境条件进行激活或停用。 在实际应用中,升级Spring Beans版本可能会带来新...

    spring3注解详细

    在使用 @Autowired 时,需要在配置文件中添加 `&lt;bean class="org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor" /&gt;` 来启用注解处理。 ### 1.1 多个同类型 Bean 的处理 当有多...

    spring2.5基于注解例子程序

    Spring 2.5是Spring框架的一个重要版本,它引入了大量增强的功能,特别是对注解的支持,这使得在Java应用程序中实现依赖注入和面向切面编程(AOP)变得更加简单和直观。在这个基于注解的例子程序中,我们将深入探讨...

    spring bean的生命周期

    了解并掌握Spring Bean的生命周期,可以帮助我们更好地设计和管理应用程序,优化Bean的创建和销毁时机,以及合理地处理Bean间的依赖关系。在实际开发中,可以利用生命周期回调方法进行一些初始化和清理工作,提高...

    Java Bean 遍历和使用注解给Bean的属性添加属性值

    - **Spring Framework**:Spring提供了`org.springframework.beans.BeanWrapper`和`org.springframework.beans.PropertyAccessor`接口,简化了对Bean的操作,包括注解的处理。 在实际开发中,遍历Java Bean和利用...

    spring-beans源码

    Bean可以定义初始化和销毁方法,也可以实现`InitializingBean`和`DisposableBean`接口,或者使用`@PostConstruct`和`@PreDestroy`注解来标记生命周期方法。 6. **Bean的作用域** Spring支持五种Bean的作用域:单例...

    spring使用注解依赖jar

    - spring-beans:包含bean工厂和bean定义,是Spring框架的基础。 - spring-aop:提供了面向切面编程的支持。 - spring-web:用于构建Web应用程序,包含Servlet监听器和MVC相关组件。 为了引入Spring框架及其注解...

    Spring框架xml注解配置方式实例

    例如,可能会有一个`@Component`注解的类表示一个Spring Bean,或者一个`@Service`、`@Repository`或`@Controller`注解的类,分别对应服务层、数据访问层和控制器层。另外,类中可能还有`@Autowired`注解来自动装配...

    Spring启动后获取所有拥有特定注解的Bean实例代码

    在ApplicationListener的实现类中,可以通过event.getApplicationContext().getBeansWithAnnotation获取到所有拥有特定注解的Beans集合,然后遍历所有bean实现业务场景。这种方式可以实现系统参数的初始化,获取系统...

    Spring注解注入属性

    这个处理器负责解析`@Autowired`注解,并在Spring容器初始化bean时执行自动装配操作。通常,只需要在配置文件中声明如下bean即可: ```xml &lt;bean class="org.springframework.beans.factory.annotation....

    spring bean XML配置入门

    Spring容器通过XML配置文件或注解来定义Bean及其相互关系。 3. **XML配置文件**: "spring-beans"是Spring中用于配置Bean的XML文件。该文件包含了一系列的&lt;bean&gt;元素,每个元素表示一个Java对象的定义,包括类名、...

    Spring系列之Spring常用注解总结1

    此外,Spring还提供了其他一些常用的注解,如@Service、@Repository和@Controller,它们分别用于标记业务层、数据访问层和控制层的bean。这些注解配合@Component,可以实现基于组件扫描的自动化配置,进一步减少.xml...

    spring beans dtd.zip

    例如,我们可以在C#代码中声明一个接口,然后在XML配置文件中定义该接口的实现类,并通过`&lt;bean&gt;`元素创建bean实例,通过`&lt;property&gt;`设置属性值,实现依赖注入。 此外,Spring.NET还支持C#的特性注解,开发者可以...

    Spring MVC非注解测试

    3. **Controller**:在非注解测试中,我们需要在XML配置文件中声明Controller类,然后定义处理特定请求的`&lt;bean&gt;`。 4. **HandlerMapping**:这个组件负责将请求映射到相应的处理器方法。 5. **HandlerAdapter**:...

Global site tag (gtag.js) - Google Analytics