<context:annotation-config />标签意义: 该标签隐式的向Spring容器注册了下面四个BeanPostProcessor: AutowiredAnnotationBeanPostProcessor CommondAnnotationBeanPostProcessor PersistenceAnnotationBeanPostProcessor RequiredAnnotationBeanPostProcessor. 1.AutowiredAnnotationBeanPostProcessor:主要是使用@AutoWired注解时,必须提前向Spring容器注册这个BeanPostprocessor 传统的配置方式:<bean class="org.springframework.beans.factory.annotation. AutowiredAnnotationBeanPostProcessor "/> 2.CommondAnnotationBeanPostprocessor:主要是使用@Resource、@PostConstruct、@Predestory等注解时,必须提前向Spring容器注册。 传统配置方式:<bean class="org.springframework.beans.factory.annotation.CommondAnnotationBeanPostprocessor"/> 3.PersistenceAnnotationBeanPostProcessor :主要是使用@PersistenceContext注解时,必须提前向Spring容器注册。 传统配置方式:<bean class="org.springframework.beans.factory.annotation.PersistenceAnnotationBeanPostProcessor"/> 4.RequiredAnnotationBeanPostProcessor:主要是使用@Required注解时,必须提前向Spring容器注册。 传统配置方式:<bean class="org.springframework.beans.factory.annotation.RequiredAnnotationBeanPostProcessor"/> 一般来说,@Autowired注解比较常用,传统配置方式有点繁琐,而这种隐式配置方式就可以帮我们很容易的搞定。 不过使用注解,我们一般都会配置扫描包路径:<context: component-scan base-package=com.xxx.xxx /> 其实,给配置已经包含了自动注入上述Processor的功能,所以,配置了包扫描之后,<context annotation-config />就可以移除了。 <context:component-scan>除了具有<context:annotation-config>的功能之外,<context:component-scan>还可以在指定的package下扫描以及注册javabean 。 下面我们通过例子来详细查看他们的区别,有三个class A,B,C,并且B,C的对象被注入到A中. package com.xxx; public class B { public B() { System.out.println("creating bean B: " + this); } } package com.xxx; public class C { public C() { System.out.println("creating bean C: " + this); } } package com.yyy; import com.xxx.B; import com.xxx.C; public class A { private B bbb; private C ccc; public A() { System.out.println("creating bean A: " + this); } public void setBbb(B bbb) { System.out.println("setting A.bbb with " + bbb); this.bbb = bbb; } public void setCcc(C ccc) { System.out.println("setting A.ccc with " + ccc); this.ccc = ccc; } }
在applicationContext.xml中加入下面的配置 : <bean id="bBean"class="com.xxx.B"/> <bean id="cBean"class="com.xxx.C"/> <bean id="aBean"class="com.yyy.A"> <property name="bbb" ref="bBean"/> <property name="ccc" ref="cBean"/> </bean>
加载applicationContext.xml配置文件,将得到下面的结果: creating bean B: com.xxx.B@c2ff5 OK, 这个结果没什么好说的,就是完全通过xml的方式,不过太过时了,下面通过注解的方式来简化我们的xml配置文件
下边看看怎么使用<context:annotation-config>和 <context:component-scan> 首先,我们使用autowire的方式将对象bbb和ccc注入到A中: package com.yyy; import org.springframework.beans.factory.annotation.Autowired; import com.xxx.B; import com.xxx.C; public class A { private B bbb; private C ccc; public A() { System.out.println("creating bean A: " + this); } @Autowired public void setBbb(B bbb) { System.out.println("setting A.bbb with " + bbb); this.bbb = bbb; } @Autowired public void setCcc(C ccc) { System.out.println("setting A.ccc with " + ccc); this.ccc = ccc; } }
然后applicationContext.xml配置文件去除属性<property>就简化为下面的样子了 <bean id="bBean"class="com.xxx.B"/> <bean id="cBean"class="com.xxx.C"/> <bean id="aBean"class="com.yyy.A"/>
当我们加载applicationContext.xml配置文件之后,将得到下面的结果: creating bean B: com.xxx.B@5e5a50
OK, ClassA中显然没有注入属性,结果是错误的的,究竟是因为什么呢?为什么我们的属性没有被注入进去呢? 是因为注解本身并不能够做任何事情,它们只是最基本的组成部分,我们需要能够处理这些注解的处理工具来处理这些注解。 这就是<context:annotation-config> 所做的事情,用于激活那些已经在spring容器里注册过的bean 我们将applicationContext.xml配置文件作如下修改: <context:annotation-config /> <bean id="bBean"class="com.xxx.B"/> <bean id="cBean"class="com.xxx.C"/> <bean id="aBean"class="com.yyy.A"/>
这回,当我们加载applicationContext.xml配置文件之后,将得到下面的结果: creating bean B: com.xxx.B@15663a2
OK, 结果正确了 但是如果我们将代码作如下修改: package com.xxx; import org.springframework.stereotype.Component; @Component public class B { public B() { System.out.println("creating bean B: " + this); } } package com.xxx; import org.springframework.stereotype.Component; @Component public class C { public C() { System.out.println("creating bean C: " + this); } } package com.yyy; import org.springframework.beans.factory.annotation.Autowired; import org.springframework.stereotype.Component; import com.xxx.B; import com.xxx.C; @Component public class A { private B bbb; private C ccc; public A() { System.out.println("creating bean A: " + this); } @Autowired public void setBbb(B bbb) { System.out.println("setting A.bbb with " + bbb); this.bbb = bbb; } @Autowired public void setCcc(C ccc) { System.out.println("setting A.ccc with " + ccc); this.ccc = ccc; } }
applicationContext.xml配置文件修改为: <context:annotation-config />
当我们加载applicationContext.xml配置文件之后,却没有任何输出,这是为什么呢? 那是因为<context:annotation-config />仅能够在已经在已经注册过的bean上面起作用。 对于没有在spring容器中注册的bean,它并不能执行任何操作。 但是不用担心,<context:component-scan> 我们将applicationContext.xml配置文件作如下修改: <context:component-scan base-package="com.xxx"/> 当我们加载applicationContext.xml的时候,会得到下面的结果: creating bean B: com.xxx.B@1be0f0a
这是什么原因呢? 是因为我们仅仅扫描了com.xxx包及其子包的类,而class A是在com.yyy包下,所以就扫描不到了 下面我们在applicationContext.xml中把com.yyy也加入进来: <context:component-scan base-package="com.xxx"/> <context:component-scan base-package="com.xxx,com.yyy"/>
然后加载applicationContext.xml就会得到下面的结果: creating bean B: com.xxx.B@cd5f8b
哇,结果正确啦 ! 回头看下我们的applicationContext.xml文件,已经简化为两行context:component-scan了,是不是很简单?
那如果我们在applicationContext.xml手动加上下面的配置, 也就是说既在applicationContext.xml中手动的注册了A的实例对象,同时,通过component-scan去扫描并注册B,C的对象,如下 <context:component-scan base-package="com.xxx"/> <bean id="aBean"class="com.yyy.A"/>
结果仍是正确的: creating bean B: com.xxx.B@157aa53
虽然class A并不是通过扫描的方式注册到容器中的 , 但是<context:component-scan> 所产生的的处理那些注解的处理器工具,会处理所有绑定到容器上面的bean,不管是通过xml手动注册的还是通过scanning扫描注册的。
那么,如果我们通过下面的方式呢?我们既配置了<context:annotation-config />,又配置了<context:component-scan base-package="com.xxx" />,它们都具有处理在容器中注册的bean里面的注解的功能。会不会出现重复注入的情况呢? <context:annotation-config /> <context:component-scan base-package="com.xxx"/> <bean id="aBean"class="com.yyy.A"/>
不用担心,不会出现的,结果如下: creating bean B: com.xxx.B@157aa53
因为<context:annotation-config />和 <context:component-scan>同时存在的时候,前者会被忽略。 也就是那些@autowire,@resource等注入注解只会被注入一次 哪怕是你手动的注册了多个处理器,spring仍然只会处理一次: <context:annotation-config /> <context:component-scan base-package="com.xxx" /> <bean id="aBean" class="com.yyy.A" /> <bean id="bla" class="org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor" /> <bean id="bla1" class="org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor" /> <bean id="bla2" class="org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor" /> <bean id="bla3" class="org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor" /> 结果仍是正确的: creating bean B: com.xxx.B@157aa53
|
|