现在的位置: 首页 > 综合 > 正文

(转)使用Spring2.5的Autowired实现注释型的IOC

2012年01月10日 ⁄ 综合 ⁄ 共 13187字 ⁄ 字号 评论关闭
  1. 使用Spring2.5的Autowired实现注释型的IOC   
  2. 阅读(31) 评论(0) 发表时间:20090116日 11:57    
  3. 本文地址:http://qzone.qq.com/blog/55357655-1232078233    
  4.   
  5. 本文标签: Autowired context IOC xml beans   
  6.     
  7.     
  8.   使用Spring2.5的新特性——Autowired可以实现快速的自动注入,而无需在xml文档里面添加bean的声明,大大减少了xml文档的维护。(偶喜欢这个功能,因为偶对xml不感冒)。       以下是一个例子:   
  9. 先编写接口Man:   
  10.        public interface Man {   
  11.            public String sayHello();   
  12. }   
  13. 然后写Man的实现类Chinese和American:   
  14.        @Service  
  15. public class Chinese implements Man{   
  16.     public String sayHello() {   
  17.         return "I am Chinese!";   
  18.     }   
  19. }   
  20.   
  21.        @Service  
  22. public class American implements Man{   
  23.     public String sayHello() {   
  24.         return "I am American!";   
  25.     }   
  26. }   
  27.   
  28. @Service注释表示定义一个bean,自动根据bean的类名实例化一个首写字母为小写的bean,例如Chinese实例化为chinese,American实例化为american,如果需要自己改名字则:@Service("你自己改的bean名")。   
  29.   
  30. beans.xml   
  31. <?xml version="1.0" encoding="UTF-8"?>   
  32. <beans xmlns="http://www.springframework.org/schema/beans"  
  33.         xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  34.         xmlns:context="http://www.springframework.org/schema/context"  
  35.         xsi:schemaLocation="http://www.springframework.org/schema/beans    
  36.            http://www.springframework.org/schema/beans/spring-beans-2.5.xsd   
  37.            http://www.springframework.org/schema/context   
  38.            http://www.springframework.org/schema/context/spring-context-2.5.xsd">   
  39.       <context:annotation-config/>   
  40.       <context:component-scan base-package="testspring.main"/>   
  41. </beans>   
  42. 在spring的配置文件里面只需要加上<context:annotation-config/>和<context:component-scan base-package="需要实现注入的类所在包"/>,可以使用base-package="*"表示全部的类。   
  43.   
  44. 编写主类测试:   
  45. @Service  
  46. public class Main {   
  47.     @Autowired  
  48.     @Qualifier("chinese")   
  49.     private Man man;   
  50.   
  51.     public static void main(String[] args) {   
  52.         // TODO code application logic here   
  53.         ApplicationContext ctx = new ClassPathXmlApplicationContext("beans.xml");   
  54.         Main main = (Main) ctx.getBean("main");   
  55.         System.out.println(main.getMan().sayHello());   
  56.     }   
  57.   
  58.     public Man getMan() {   
  59.         return man;   
  60.     }   
  61. }   
  62.   
  63. 在Man接口前面标上@Autowired@Qualifier注释使得Man接口可以被容器注入,当Man接口存在两个实现类的时候必须指定其中一个来注入,使用实现类首字母小写的字符串来注入。否则可以省略,只写@Autowired    
  64.   
  65. **********************   
  66. 使用 Spring 2.5 注释驱动的 IoC 功能   
  67. 发表于08-03-04 20:38 | 阅读 1285 | 评分 (暂无)    
  68. 概述   
  69.   
  70. 注释配置相对于 XML 配置具有很多的优势:   
  71.   
  72. 它可以充分利用 Java 的反射机制获取类结构信息,这些信息可以有效减少配置的工作。如使用 JPA 注释配置 ORM 映射时,我们就不需要指定 PO 的属性名、类型等信息,如果关系表字段和 PO 属性名、类型都一致,您甚至无需编写任务属性映射信息——因为这些信息都可以通过 Java 反射机制获取。    
  73. 注释和 Java 代码位于一个文件中,而 XML 配置采用独立的配置文件,大多数配置信息在程序开发完成后都不会调整,如果配置信息和 Java 代码放在一起,有助于增强程序的内聚性。而采用独立的 XML 配置文件,程序员在编写一个功能时,往往需要在程序文件和配置文件中不停切换,这种思维上的不连贯会降低开发效率。    
  74. 因此在很多情况下,注释配置比 XML 配置更受欢迎,注释配置有进一步流行的趋势。Spring 2.5 的一大增强就是引入了很多注释类,现在您已经可以使用注释配置完成大部分 XML 配置的功能。在这篇文章里,我们将向您讲述使用注释进行 Bean 定义和依赖注入的内容。   
  75.   
  76.   
  77.   
  78.     
  79.   
  80.   
  81.  回页首    
  82.     
  83.   
  84.   
  85.   
  86. 原来我们是怎么做的   
  87.   
  88. 在使用注释配置之前,先来回顾一下传统上是如何配置 Bean 并完成 Bean 之间依赖关系的建立。下面是 3 个类,它们分别是 Office、Car 和 Boss,这 3 个类需要在 Spring 容器中配置为 Bean:   
  89.   
  90. Office 仅有一个属性:   
  91.   
  92.   
  93. 清单 1. Office.java   
  94.                    
  95. package com.baobaotao;   
  96. public class Office {   
  97.     private String officeNo =”001”;   
  98.   
  99.     //省略 get/setter   
  100.   
  101.     @Override  
  102.     public String toString() {   
  103.         return "officeNo:" + officeNo;   
  104.     }   
  105. }   
  106.     
  107.   
  108.   
  109. Car 拥有两个属性:   
  110.   
  111.   
  112. 清单 2. Car.java   
  113.                    
  114. package com.baobaotao;   
  115.   
  116. public class Car {   
  117.     private String brand;   
  118.     private double price;   
  119.   
  120.     // 省略 get/setter   
  121.   
  122.     @Override  
  123.     public String toString() {   
  124.         return "brand:" + brand + "," + "price:" + price;   
  125.     }   
  126. }   
  127.     
  128.   
  129.   
  130. Boss 拥有 Office 和 Car 类型的两个属性:   
  131.   
  132.   
  133. 清单 3. Boss.java   
  134.                    
  135. package com.baobaotao;   
  136.   
  137. public class Boss {   
  138.     private Car car;   
  139.     private Office office;   
  140.   
  141.     // 省略 get/setter   
  142.   
  143.     @Override  
  144.     public String toString() {   
  145.         return "car:" + car + "/n" + "office:" + office;   
  146.     }   
  147. }   
  148.     
  149.   
  150.   
  151. 我们在 Spring 容器中将 Office 和 Car 声明为 Bean,并注入到 Boss Bean 中:下面是使用传统 XML 完成这个工作的配置文件 beans.xml:   
  152.   
  153.   
  154. 清单 4. beans.xml 将以上三个类配置成 Bean   
  155.                    
  156. <?xml version="1.0" encoding="UTF-8" ?>   
  157. <beans xmlns="http://www.springframework.org/schema/beans"  
  158.     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  159.     xsi:schemaLocation="http://www.springframework.org/schema/beans    
  160.  http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">   
  161.     <bean id="boss" class="com.baobaotao.Boss">   
  162.         <property name="car" ref="car"/>   
  163.         <property name="office" ref="office" />   
  164.     </bean>   
  165.     <bean id="office" class="com.baobaotao.Office">   
  166.         <property name="officeNo" value="002"/>   
  167.     </bean>   
  168.     <bean id="car" class="com.baobaotao.Car" scope="singleton">   
  169.         <property name="brand" value=" 红旗 CA72"/>   
  170.         <property name="price" value="2000"/>   
  171.     </bean>   
  172. </beans>   
  173.     
  174.   
  175.   
  176. 当我们运行以下代码时,控制台将正确打出 boss 的信息:   
  177.   
  178.   
  179. 清单 5. 测试类:AnnoIoCTest.java   
  180.                    
  181. import org.springframework.context.ApplicationContext;   
  182. import org.springframework.context.support.ClassPathXmlApplicationContext;   
  183. public class AnnoIoCTest {   
  184.   
  185.     public static void main(String[] args) {   
  186.         String[] locations = {"beans.xml"};   
  187.         ApplicationContext ctx =    
  188.             new ClassPathXmlApplicationContext(locations);   
  189.         Boss boss = (Boss) ctx.getBean("boss");   
  190.         System.out.println(boss);   
  191.     }   
  192. }   
  193.     
  194.   
  195.   
  196. 这说明 Spring 容器已经正确完成了 Bean 创建和装配的工作。   
  197.   
  198.   
  199.   
  200.     
  201.   
  202.   
  203.  回页首    
  204.     
  205.   
  206.   
  207.   
  208. 使用 @Autowired 注释   
  209.   
  210. Spring 2.5 引入了 @Autowired 注释,它可以对类成员变量、方法及构造函数进行标注,完成自动装配的工作。来看一下使用 @Autowired 进行成员变量自动注入的代码:   
  211.   
  212.   
  213. 清单 6. 使用 @Autowired 注释的 Boss.java   
  214.                    
  215. package com.baobaotao;   
  216. import org.springframework.beans.factory.annotation.Autowired;   
  217.   
  218. public class Boss {   
  219.   
  220.     @Autowired  
  221.     private Car car;   
  222.   
  223.     @Autowired  
  224.     private Office office;   
  225.   
  226.     …   
  227. }   
  228.     
  229.   
  230.   
  231. Spring 通过一个 BeanPostProcessor 对 @Autowired 进行解析,所以要让 @Autowired 起作用必须事先在 Spring 容器中声明 AutowiredAnnotationBeanPostProcessor Bean。   
  232.   
  233.   
  234. 清单 7. 让 @Autowired 注释工作起来   
  235.                    
  236. <?xml version="1.0" encoding="UTF-8" ?>   
  237. <beans xmlns="http://www.springframework.org/schema/beans"  
  238.     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  239.     xsi:schemaLocation="http://www.springframework.org/schema/beans    
  240.  http://www.springframework.org/schema/beans/spring-beans-2.5.xsd">   
  241.   
  242.     <!-- 该 BeanPostProcessor 将自动起作用,对标注 @Autowired 的 Bean 进行自动注入 -->   
  243.     <bean class="org.springframework.beans.factory.annotation.   
  244.         AutowiredAnnotationBeanPostProcessor"/>   
  245.   
  246.     <!-- 移除 boss Bean 的属性注入配置的信息 -->   
  247.     <bean id="boss" class="com.baobaotao.Boss"/>   
  248.     
  249.     <bean id="office" class="com.baobaotao.Office">   
  250.         <property name="officeNo" value="001"/>   
  251.     </bean>   
  252.     <bean id="car" class="com.baobaotao.Car" scope="singleton">   
  253.         <property name="brand" value=" 红旗 CA72"/>   
  254.         <property name="price" value="2000"/>   
  255.     </bean>   
  256. </beans>   
  257.     
  258.   
  259.   
  260. 这样,当 Spring 容器启动时,AutowiredAnnotationBeanPostProcessor 将扫描 Spring 容器中所有 Bean,当发现 Bean 中拥有 @Autowired 注释时就找到和其匹配(默认按类型匹配)的 Bean,并注入到对应的地方中去。   
  261.   
  262. 按照上面的配置,Spring 将直接采用 Java 反射机制对 Boss 中的 car 和 office 这两个私有成员变量进行自动注入。所以对成员变量使用 @Autowired 后,您大可将它们的 setter 方法(setCar() 和 setOffice())从 Boss 中删除。   
  263.   
  264. 当然,您也可以通过 @Autowired 对方法或构造函数进行标注,来看下面的代码:   
  265.   
  266.   
  267. 清单 8. 将 @Autowired 注释标注在 Setter 方法上   
  268.                    
  269. package com.baobaotao;   
  270.   
  271. public class Boss {   
  272.     private Car car;   
  273.     private Office office;   
  274.   
  275.      @Autowired  
  276.     public void setCar(Car car) {   
  277.         this.car = car;   
  278.     }   
  279.     
  280.     @Autowired  
  281.     public void setOffice(Office office) {   
  282.         this.office = office;   
  283.     }   
  284.     …   
  285. }   
  286.     
  287.   
  288.   
  289. 这时,@Autowired 将查找被标注的方法的入参类型的 Bean,并调用方法自动注入这些 Bean。而下面的使用方法则对构造函数进行标注:   
  290.   
  291.   
  292. 清单 9. 将 @Autowired 注释标注在构造函数上   
  293.                    
  294. package com.baobaotao;   
  295.   
  296. public class Boss {   
  297.     private Car car;   
  298.     private Office office;   
  299.     
  300.     @Autowired  
  301.     public Boss(Car car ,Office office){   
  302.         this.car = car;   
  303.         this.office = office ;   
  304.     }   
  305.     
  306.     …   
  307. }   
  308.     
  309.   
  310.   
  311. 由于 Boss() 构造函数有两个入参,分别是 car 和 office,@Autowired 将分别寻找和它们类型匹配的 Bean,将它们作为 Boss(Car car ,Office office) 的入参来创建 Boss Bean。   
  312.   
  313.   
  314.   
  315.     
  316.   
  317.   
  318.  回页首    
  319.     
  320.   
  321.   
  322.   
  323. 当候选 Bean 数目不为 1 时的应对方法   
  324.   
  325. 在默认情况下使用 @Autowired 注释进行自动注入时,Spring 容器中匹配的候选 Bean 数目必须有且仅有一个。当找不到一个匹配的 Bean 时,Spring 容器将抛出 BeanCreationException 异常,并指出必须至少拥有一个匹配的 Bean。我们可以来做一个实验:   
  326.   
  327.   
  328. 清单 10. 候选 Bean 数目为 0 时   
  329.                    
  330. <?xml version="1.0" encoding="UTF-8" ?>   
  331. <beans xmlns="http://www.springframework.org/schema/beans"  
  332.     xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"  
  333.      xsi:schemaLocation="http://www.springframework.org/schema/beans    
  334.  http://www.springframework.org/schema/beans/spring-beans-2.5.xsd ">   
  335.     
  336.     <bean class="org.springframework.beans.factory.annotation.   
  337.         AutowiredAnnotationBeanPostProcessor"/>    
  338.   
  339.     <bean id="boss" class="com.baobaotao.Boss"/>   
  340.   
  341.     <!-- 将 office Bean 注释掉 -->   
  342.     <!-- <bean id="office" class="com.baobaotao.Office">   
  343.     <property name="officeNo" value="001"/>   
  344.     </bean>-->   
  345.   
  346.     <bean id="car" class="com.baobaotao.Car" scope="singleton">   
  347.         <property name="brand" value=" 红旗 CA72"/>   
  348.         <property name="price" value="2000"/>   
  349.     </bean>   
  350. </beans>   
  351.     
  352.   
  353.   
  354. 由于 office Bean 被注释掉了,所以 Spring 容器中将没有类型为 Office 的 Bean 了,而 Boss 的 office 属性标注了 @Autowired,当启动 Spring 容器时,异常就产生了。   
  355.   
  356. 当不能确定 Spring 容器中一定拥有某个类的 Bean 时,可以在需要自动注入该类 Bean 的地方可以使用 @Autowired(required = false),这等于告诉 Spring:在找不到匹配 Bean 时也不报错。来看一下具体的例子:   
  357.   
  358.   
  359. 清单 11. 使用 @Autowired(required = false)   
  360.                    
  361. package com.baobaotao;   
  362.   
  363. import org.springframework.beans.factory.annotation.Autowired;   
  364. import org.springframework.beans.factory.annotation.Required;   
  365.   
  366. public class Boss {   
  367.   
  368.     private Car car;   
  369.     private Office office;   
  370.   
  371.     @Autowired  
  372.     public void setCar(Car car) {   
  373.         this.car = car;   
  374.     }   
  375.     @Autowired(required = false)   
  376.     public void setOffice(Office office) {   
  377.         this.office = office;   
  378.     }   
  379.     …   
  380. }   
  381.     
  382.   
  383.   
  384. 当然,一般情况下,使用 @Autowired 的地方都是需要注入 Bean 的,使用了自动注入而又允许不注入的情况一般仅会在开发期或测试期碰到(如为了快速启动 Spring 容器,仅引入一些模块的 Spring 配置文件),所以 @Autowired(required = false) 会很少用到。   
  385.   
  386. 和找不到一个类型匹配 Bean 相反的一个错误是:如果 Spring 容器中拥有多个候选 Bean,Spring 容器在启动时也会抛出 BeanCreationException 异常。来看下面的例子:   
  387.   
  388.   
  389. 清单 12. 在 beans.xml 中配置两个 Office 类型的 Bean   
  390.                    
  391. …    
  392. <bean id="office" class="com.baobaotao.Office">   
  393.     <property name="officeNo" value="001"/>   
  394. </bean>   
  395. <bean id="office2" class="com.baobaotao.Office">   
  396.     <property name="officeNo" value="001"/>   
  397. </bean>   
  398. …   
  399.     
  400.   
  401.   
  402. 我们在 Spring 容器中配置了两个类型为 Office 类型的 Bean,当对 Boss 的 office 成员变量进行自动注入时,Spring 容器将无法确定到底要用哪一个 Bean,因此异常发生了。   
  403.   
  404. Spring 允许我们通过 @Qualifier 注释指定注入 Bean 的名称,这样歧义就消除了,可以通过下面的方法解决异常:   
  405.   
  406.   
  407. 清单 13. 使用 @Qualifier 注释指定注入 Bean 的名称   
  408.                    
  409. @Autowired  
  410. public void setOffice(@Qualifier("office")Office office) {   
  411.     this.office = office;   
  412. }   
  413.     
  414.   
  415.   
  416. @Qualifier("office") 中的 office 是 Bean 的名称,所以 @Autowired 和 @Qualifier 结合使用时,自动注入的策略就从 byType 转变成 byName 了。@Autowired 可以对成员变量、方法以及构造函数进行注释,而 @Qualifier 的标注对象是成员变量、方法入参、构造函数入参。正是由于注释对象的不同,所以 Spring 不将 @Autowired 和 @Qualifier 统一成一个注释类。下面是对成员变量和构造函数入参进行注释的代码:   
  417.   
  418. 对成员变量进行注释:   
  419.   
  420.   
  421. 清单 14. 对成员变量使用 @Qualifier 注释   
  422.                    
  423. public class Boss {   
  424.     @Autowired  
  425.     private Car car;   
  426.     
  427.     @Autowired  
  428.     @Qualifier("office")   
  429.     private Office office;   
  430.     …   
  431. }   
  432.     
  433.   
  434.   
  435. 对构造函数入参进行注释:   
  436.   
  437.   
  438. 清单 15. 对构造函数变量使用 @Qualifier 注释   
  439.                    
  440. public class Boss {   
  441.     private Car car;   
  442.     private Office office;   
  443.   
  444.     @Autowired  
  445.     public Boss(Car car , @Qualifier("office")Office office){   
  446.         this.car = car;   
  447.         this.office = office ;   
  448.     }   
  449. }   
  450.     
  451.   
  452.   
  453. @Qualifier 只能和 @Autowired 结合使用,是对 @Autowired 有益的补充。一般来讲,@Qualifier 对方法签名中入参进行注释会降低代码的可读性,而对成员变量注释则相对好一些。   
  454.   
  455.   
  456.   
  457.     
  458.   
  459.   
  460.  回页首    
  461.     
  462.   
  463.   
  464.   
  465. 使用 JSR-250 的注释   
  466.   
  467. Spring 不但支持自己定义的 @Autowired 的注释,还支持几个由 JSR-250 规范定义的注释,它们分别是 @Resource@PostConstruct 以及 @PreDestroy。   
  468.   
  469. @Resource  
  470.   
  471. @Resource 的作用相当于 @Autowired,只不过 @Autowired 按 byType 自动注入,面 @Resource 默认按 byName 自动注入罢了。@Resource 有两个属性是比较重要的,分别是 name 和 type,Spring 将 @Resource 注释的 name 属性解析为 Bean 的名字,而 type 属性则解析为 Bean 的类型。所以如果使用 name 属性,则使用 byName 的自动注入策略,而使用 type 属性时则使用 byType 自动注入策略。如果既不指定 name 也不指定 type 属性,这时将通过反射机制使用 byName 自动注入策略。   
  472.   
  473. Resource 注释类位于 Spring 发布包的 lib/j2ee/common-annotations.jar 类包中,因此在使用之前必须将其加入到项目的类库中。来看一个使用 @Resource 的例子:   
  474.   
  475.   
  476. 清单 16. 使用 @Resource 注释的 Boss.java   
  477.                    
  478. package com.baobaotao;   
  479.   
  480. import javax.annotation.Resource;   
  481.   
  482. public class Boss {   
  483.     // 自动注入类型为 Car 的 Bean   
  484.     @Resource  
  485.     private Car car;   
  486.   
  487.     // 自动注入 bean 名称为 office 的 Bean   
  488.     @Resource(name = "office")   
  489.     private Office office;   
  490. }   
  491.     
  492.   
  493.   
  494. 一般情况下,我们无需使用类似于 @Resource(type=Car.class) 的注释方式,因为 Bean 的类型信息可以通过 Java 反射从代码中获取。   
  495.   
  496. 要让 JSR-250 的注释生效,除了在 Bean 类中标注这些注释外,还需要在 Spring 容器中注册一个负责处理这些注释的 BeanPostProcessor:   
  497.   
  498. <bean    
  499.   class="org.springframework.context.annotation.CommonAnnotationBeanPostProcessor"/>   
  500.     
  501.   
  502.   
  503. CommonAnnotationBeanPostProcessor 实现了 BeanPostProcessor 接口,它负责扫描使用了 JSR-250 注释的 Bean,并对它们进行相应的操作。   
  504.   
  505. @PostConstruct 和 @PreDestroy  
  506.   
  507. Spring 容器中的 Bean 是有生命周期的,Spring 允许在 Bean 在初始化完成后以及 Bean 销毁前执行特定的操作,您既可以通过实现 InitializingBean/DisposableBean 接口来定制初始化之后 / 销毁之前的操作方法,也可以通过 <bean> 元素的 init-method/destroy-method 属性指定初始化之后 / 销毁之前调用的操作方法。关于 Spring 的生命周期,笔者在《精通 Spring 2.x—企业应用开发精解》第 3 章进行了详细的描述,有兴趣的读者可以查阅。   
  508.   
  509. JSR-250 为初始化之后/销毁之前方法的指定定义了两个注释类,分别是 @PostConstruct 和 @PreDestroy,这两个注释只能应用于方法上。标注了 @PostConstruct 注释的方法将在类实例化后调用,而标注了 @PreDestroy 的方法将在类销毁之前调用。   
  510.   
  511.   
  512. <

【上篇】
【下篇】

抱歉!评论已关闭.