转自:GOOD spring <context:annotation-config> 跟 <context:component-scan>诠释及区别

<context:annotation-config>  是用于激活那些已经在spring容器里注册过的bean(无论是通过xml的方式还是通过package sanning的方式)上面的注解,是一个注解处理工具,也就是只管注入,不管注册bean。

<context:component-scan> 除了具有<context:annotation-config>的功能之外,<context:component-scan>还可以在指定的package下扫描以及注册javabean ,也就是既管注入,又管注册bean。

下面我们通过例子来详细查看他们的区别,

有三个class   A,B,C,并且B,C的对象被注入到A中.

  1. package com.xxx;
  2. public class B {
  3. public B() {
  4. System.out.println("creating bean B: " + this);
  5. }
  6. }
  7.  
  8. package com.xxx;
  9. public class C {
  10. public C() {
  11. System.out.println("creating bean C: " + this);
  12. }
  13. }
  14.  
  15. package com.yyy;
  16. import com.xxx.B;
  17. import com.xxx.C;
  18. public class A {
  19. private B bbb;
  20. private C ccc;
  21. public A() {
  22. System.out.println("creating bean A: " + this);
  23. }
  24. public void setBbb(B bbb) {
  25. System.out.println("setting A.bbb with " + bbb);
  26. this.bbb = bbb;
  27. }
  28. public void setCcc(C ccc) {
  29. System.out.println("setting A.ccc with " + ccc);
  30. this.ccc = ccc;
  31. }
  32. }  

在applicationContext.xml中加入下面的配置 :

  1. <bean id="bBean"class="com.xxx.B"/>
  2. <bean id="cBean"class="com.xxx.C"/>
  3. <bean id="aBean"class="com.yyy.A">
  4. <property name="bbb" ref="bBean"/>
  5. <property name="ccc" ref="cBean"/>
  6. </bean>

加载applicationContext.xml配置文件,将得到下面的结果:

  1. creating bean B: com.xxx.B@c2ff5
  2. creating bean C: com.xxx.C@1e8a1f6
  3. creating bean A: com.yyy.A@1e152c5
  4. setting A.bbb with com.xxx.B@c2ff5
  5. setting A.ccc with com.xxx.C@1e8a1f6

OK, 这个结果没什么好说的,就是完全通过xml的方式,不过太过时了,下面通过注解的方式来简化我们的xml配置文件

首先,我们使用autowire的方式将对象bbb和ccc注入到A中:

  1. package com.yyy;
  2. import org.springframework.beans.factory.annotation.Autowired;
  3. import com.xxx.B;
  4. import com.xxx.C;
  5. public class A {
  6. private B bbb;
  7. private C ccc;
  8. public A() {
  9. System.out.println("creating bean A: " + this);
  10. }
  11. @Autowired
  12. public void setBbb(B bbb) {
  13. System.out.println("setting A.bbb with " + bbb);
  14. this.bbb = bbb;
  15. }
  16. @Autowired
  17. public void setCcc(C ccc) {
  18. System.out.println("setting A.ccc with " + ccc);
  19. this.ccc = ccc;
  20. }
  21. }  

然后,我们就可以从applicationContext.xml中移除下面的配置

  1. <property name="bbb" ref="bBean"/>
  2. <property name="ccc" ref="cBean"/>

移除之后,我们的applicationContext.xml配置文件就简化为下面的样子了

  1. <bean id="bBean"class="com.xxx.B"/>
  2. <bean id="cBean"class="com.xxx.C"/>
  3. <bean id="aBean"class="com.yyy.A"/>

当我们加载applicationContext.xml配置文件之后,将得到下面的结果:

  1. creating bean B: com.xxx.B@5e5a50
  2. creating bean C: com.xxx.C@54a328
  3. creating bean A: com.yyy.A@a3d4cf

OK, 结果是错误的的,究竟是因为什么呢?为什么我们的属性没有被注入进去呢?

是因为注解本身并不能够做任何事情,它们只是最基本的组成部分,我们需要能够处理这些注解的处理工具来处理这些注解

这就是<context:annotation-config> 所做的事情

我们将applicationContext.xml配置文件作如下修改:

  1. <context:annotation-config />
  2. <bean id="bBean"class="com.xxx.B"/>
  3. <bean id="cBean"class="com.xxx.C"/>
  4. <bean id="aBean"class="com.yyy.A"/>

当我们加载applicationContext.xml配置文件之后,将得到下面的结果:

  1. creating bean B: com.xxx.B@15663a2
  2. creating bean C: com.xxx.C@cd5f8b
  3. creating bean A: com.yyy.A@157aa53
  4. setting A.bbb with com.xxx.B@15663a2
  5. setting A.ccc with com.xxx.C@cd5f8b

OK, 结果正确了。

下面演示<context:annotation-config> 跟 <context:component-scan>区别:

但是如果我们将代码作如下修改:

  1. package com.xxx;
  2. import org.springframework.stereotype.Component;
  3. @Component
  4. public class B {
  5. public B() {
  6. System.out.println("creating bean B: " + this);
  7. }
  8. }
  9.  
  10. package com.xxx;
  11. import org.springframework.stereotype.Component;
  12. @Component
  13. public class C {
  14. public C() {
  15. System.out.println("creating bean C: " + this);
  16. }
  17. }
  18.  
  19. package com.yyy;
  20. import org.springframework.beans.factory.annotation.Autowired;
  21. import org.springframework.stereotype.Component;
  22. import com.xxx.B;
  23. import com.xxx.C;
  24. @Component
  25. public class A {
  26. private B bbb;
  27. private C ccc;
  28. public A() {
  29. System.out.println("creating bean A: " + this);
  30. }
  31. @Autowired
  32. public void setBbb(B bbb) {
  33. System.out.println("setting A.bbb with " + bbb);
  34. this.bbb = bbb;
  35. }
  36. @Autowired
  37. public void setCcc(C ccc) {
  38. System.out.println("setting A.ccc with " + ccc);
  39. this.ccc = ccc;
  40. }
  41. }

applicationContext.xml配置文件修改为:

  1. <context:annotation-config />

当我们加载applicationContext.xml配置文件之后,却没有任何输出,这是为什么呢?

那是因为<context:annotation-config />仅能够在已经在已经注册过的bean上面起作用。

对于没有在spring容器中注册的bean,它并不能执行任何操作。

但是不用担心,<context:component-scan>除了具有<context:annotation-config />的功能之外,还具有自动将带有@component,@service,@Repository等注解的对象注册到spring容器中的功能。

我们将applicationContext.xml配置文件作如下修改:

  1. <context:component-scan base-package="com.xxx"/>

当我们加载applicationContext.xml的时候,会得到下面的结果:

  1. creating bean B: com.xxx.B@1be0f0a
  2. creating bean C: com.xxx.C@80d1ff

这是什么原因呢?

是因为我们仅仅扫描了com.xxx包及其子包的类,而class  A是在com.yyy包下,所以就扫描不到了

下面我们在applicationContext.xml中把com.yyy也加入进来:

  1. <context:component-scan base-package="com.xxx,com.yyy"/>
  1. 然后加载applicationContext.xml就会得到下面的结果:
  1. creating bean B: com.xxx.B@cd5f8b
  2. creating bean C: com.xxx.C@15ac3c9
  3. creating bean A: com.yyy.A@ec4a87
  4. setting A.bbb with com.xxx.B@cd5f8b
  5. setting A.ccc with com.xxx.C@15ac3c9

哇,结果正确啦 !

回头看下我们的applicationContext.xml文件,已经简化为:

  1. <context:component-scan base-package="com.xxx,com.yyy"/>

了。

那如果我们在applicationContext.xml手动加上下面的配置,也就是说既在applicationContext.xml中手动的注册了A的实例对象,同时,通过component-scan去扫描并注册B,C的对象

  1. <context:component-scan base-package="com.xxx"/>
  2. <bean id="aBean"class="com.yyy.A"/>

结果仍是正确的:

  1. creating bean B: com.xxx.B@157aa53
  2. creating bean C: com.xxx.C@ec4a87
  3. creating bean A: com.yyy.A@1d64c37
  4. setting A.bbb with com.xxx.B@157aa53
  5. setting A.ccc with com.xxx.C@ec4a87

虽然class  A并不是通过扫描的方式注册到容器中的 ,但是<context:component-scan> 所产生的的处理那些注解的处理器工具,会处理所有绑定到容器上面的bean,不管是通过xml手动注册的还是通过scanning扫描注册的。

那么,如果我们通过下面的方式呢?我们既配置了<context:annotation-config />,又配置了<context:component-scan base-package="com.xxx" />,它们都具有处理在容器中注册的bean里面的注解的功能。会不会出现重复注入的情况呢?

  1. <context:annotation-config /><context:component-scan base-package="com.xxx"/><bean id="aBean"class="com.yyy.A"/>

不用担心,不会出现的:

  1. creating bean B: com.xxx.B@157aa53
  2. creating bean C: com.xxx.C@ec4a87
  3. creating bean A: com.yyy.A@1d64c37
  4. setting A.bbb with com.xxx.B@157aa53
  5. setting A.ccc with com.xxx.C@ec4a87

因为<context:annotation-config />和 <context:component-scan>同时存在的时候,前者会被忽略。也就是那些@autowire,@resource等注入注解只会被注入一次

哪怕是你手动的注册了多个处理器,spring仍然只会处理一次:

  1. <context:annotation-config />
  2. <context:component-scan base-package="com.xxx" />
  3. <bean id="aBean" class="com.yyy.A" />
  4. <bean id="bla" class="org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor" />
  5. <bean id="bla1" class="org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor" />
  6. <bean id="bla2" class="org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor" />
  7. <bean id="bla3" class="org.springframework.beans.factory.annotation.AutowiredAnnotationBeanPostProcessor" />

结果仍是正确的:

  1. creating bean B: com.xxx.B@157aa53
  2. creating bean C: com.xxx.C@ec4a87
  3. creating bean A: com.yyy.A@25d2b2
  4. setting A.bbb with com.xxx.B@157aa53
  5. setting A.ccc with com.xxx.C@ec4a87
 
 

<context:annotation-config> 和 <context:component-scan>的区别的更多相关文章

  1. <input type="button" /> 和<input type="submit" /> 的区别

    <input type="button" /> 这就是一个按钮.如果你不写javascript 的话,按下去什么也不会发生.<input type="s ...

  2. ASP.NET控件<ASP:Button /> html控件<input type="button">区别联系

    ASP.NET控件<ASP:Button />-------html控件<input type="button">杨中科是这么说的:asp和input是一样 ...

  3. <button>和<input type="button"> 的区别

    <button>标签 定义和用法 <button> 标签定义一个按钮. 在 button 元素内部,您可以放置内容,比如文本或图像.这是该元素与使用 input 元素创建的按钮 ...

  4. <button>与<input type="button">的区别

    一.定义和用法 <button> 标签定义的是一个按钮. 在 button 元素内部,可以放置文本或图像.这是<button>与使用 input 元素创建的按钮的不同之处. 二 ...

  5. 解析button和input type=”button”的区别

    一.定义和用法 <button> 标签定义的是一个按钮. 在 button 元素内部,可以放置文本或图像.这是<button>与使用 input 元素创建的按钮的不同之处. 二 ...

  6. 【转】解析<button>和<input type="button"> 的区别

    一.定义和用法 <button> 标签定义的是一个按钮. 在 button 元素内部,可以放置文本或图像.这是<button>与使用 input 元素创建的按钮的不同之处. 二 ...

  7. 了解HTML表单之input元素的23种type类型

    目录 传统类型 text password file radio checkbox hidden button image reset submit 新增类型 color tel email url ...

  8. HTML(七)HTML 表单(form元素介绍,input元素的常用type类型,input元素的常用属性)

    前言 表单是网页与用户的交互工具,由一个<form>元素作为容器构成,封装其他任何数量的表单控件,还有其他任何<body>元素里可用的标签 表单能够包含<input> ...

  9. HTML表单之input元素的23种type类型

    摘自:http://www.cnblogs.com/xiaohuochai/p/5179909.html 了解HTML表单之input元素的23种type类型 随着HTML5的出现,input元素新增 ...

  10. HTML中button和input button的区别

    button和input button的区别 一句话概括主题:<button>具有<input type="button" ... >相同的作用但是在可操控 ...

随机推荐

  1. SQLServer 重建索引前后对比

    在做维护项目的时,我们经常会遇到索引维护的问题,通过语句,我们就可以判断某个表的索引是否需要重建. 执行一下语句:先分析表的索引 分析表的索引建立情况:DBCC showcontig('Table') ...

  2. 信息存储——当值X是2的非负整数n次幂时,如何表示成十六进制

    十六进制表示法              当值X是2的非负整数n次幂时,很容易将X写成十六进制形式,只要记住X的二进制表示就是1后面跟n个0.十六进制数字0代表4个二进制0.所以当n表示成i+4j的形 ...

  3. Sql Server 表创建以及Ef浅谈

    1.在数据库中新建两张测试表 创建用户表 use eftest go if exists(select * from sysobjects where name='UserInfo') drop ta ...

  4. sf中标准的分页功能介绍

    世上本无事,庸人自扰之.我喜欢一个相对比较安静的环境去学习和工作,希望在一个掉一根针的声音都能够听到的环境中,但是有时候往往相反,一片嘈杂,我改变不了周围的环境,只能改变自己,其实这些都没有什么,也许 ...

  5. fopen()函数以"a+"方式打开一个不存在的文件后读写出现问题

    问题:在完成课后习题的时候,使用fopen()函数以"a+"方式打开一个不存在的文件时,写入.读取出现错误: //添加用户输入单词后,在单词头加入编号,确保编号跟着前面的开始排序 ...

  6. 3个微信小程序体验报告

    1.小程序摩拜单车.腾讯视频.JD的体报告 2.小程序的入口存在不公平 3.小程序2.0会怎么样?WSO浅谈 KEVIN常用的APP是以摩拜单车与JD商城和大众点评等,那么今天也就通过这上个进行对比 ...

  7. Java Web 开发环境快速搭建

    Java Web 开发环境快速搭建 在因某种原因更换开发设备后,可依据此文快速搭建开发环境,恢复工作环境. Java开发环境: Windows 10 (64-bit) Oralce JDK Eclip ...

  8. Matches Puzzle Game

    Matches Puzzle Game 题目链接:http://acm.hdu.edu.cn/showproblem.php?pid=5456 数位DP 首先我把C-A=B改为A+B=C(我觉得会简单 ...

  9. Redis断线重连编码注意事项

    应用在Redis重启.网络闪断并恢复正常后,应用必须能够自恢复,下面以Java语言的jedis客户端为例说明: 1.作为发布者 Jedis对象不能作为单例,网络闪断后该Jedis对象无法自恢复.应该每 ...

  10. iOS开发app自动更新的实现

    #define kStoreAppId @“xxxxxxxxx” // (appid数字串) -(void)checkAppUpdate { NSDictionary *infoDict = [[NS ...