`
teamojiao
  • 浏览: 350424 次
  • 性别: Icon_minigender_1
  • 来自: 南京
社区版块
存档分类
最新评论

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

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

相关推荐

    spring2.5 注解

    使用Spring2.5的Autowired实现注释型的IOC , 使用Spring2.5的新特性——Autowired可以实现快速的自动注入,而无需在xml文档里面添加bean的声明,大大减少了xml文档的维护

    使用 Spring 2_5 注释驱动的 IoC 功能.mht

    **Spring 2.5 注释驱动的 IoC 功能详解** Spring 框架自 2.5 版本开始引入了对注解的强大支持,极大地简化了依赖注入(Dependency Injection,简称 DI)的配置过程。注解驱动的 IoC(Inverse of Control,控制反转...

    spring 源码中文注释

    Spring框架是Java开发中最广泛应用的轻量级框架之一,它以IoC(Inversion of Control,控制反转)和AOP(Aspect Oriented Programming,面向切面编程)为核心,极大地简化了企业级应用的开发工作。这份"spring 源码...

    基于Maven构建的Spring IoC源码实例

    在"基于Maven构建的Spring IoC源码实例"中,我们可以学到如何使用Maven搭建Spring项目,并通过Spring IoC实现组件间的依赖注入。以下是这个实例中可能包含的关键知识点: 1. **Maven项目结构**:了解标准的Maven...

    spring-source中文注释版.zip

    下载"spring-source中文注释版.zip"后,你可以使用IDEA这样的集成开发环境直接导入,通过阅读注释来深入理解Spring框架的工作机制,这对于提升Java开发技能和解决实际问题大有裨益。同时,结合Spring官方文档和其他...

    关于spring boot中几种注入方法的一些个人看法

    在 Spring 2.5 引入了 @Autowired 注释,我们平常直接引用的时候很少注意这些,只是自己写好了一个方法或者 springboot 自动配置好的一个方法我们要在另一个类中去调用,这个时候,我们就会采用该注释。 2. @...

    spring-framework-5.2.0.RELEASE-master.zip

    通过阅读`@Autowired`、`@PostConstruct`、`@PreDestroy`等注解的实现,我们可以了解到Spring如何进行自动装配。同时,对于`BeanDefinition`、`BeanPostProcessor`等关键接口的注释,能帮助我们深入理解bean的生命...

    简单手写springIOC注解模式

    3. **@Autowired**: Spring通过此注解实现依赖注入。Spring会根据类型或名称自动匹配并注入依赖。如果需要基于名称注入,可以添加`@Qualifier("dependencyName")`。 4. **@Repository、@Service、@Controller**: ...

    spring源码深度解析(包含所有spring源码,带全部的注释以及案例,很适合入门级)

    1. **IoC容器**:Spring的核心是IoC容器,它负责管理对象的生命周期和对象之间的依赖关系。IoC通过配置文件或注解来定义对象及其依赖关系,实现对象的创建和管理。在源码中,你可以看到`BeanFactory`和`...

    ioc-aop:看完spring的源码后,自己手动实现一个类似Spring的IOC和AOP功能的演示,代码注释详细,项目可以直接运行-看

    于是经过一番研究手动实现一个类似Spring的IOC和AOP功能的演示,本demo成功实现了容器的依赖注入和切面的功能,aop使用CGLIB实现。 说明 首先代码的测试运行demo见 应用程序 public static void main(String[] args...

    spring+maven项目工程

    在"通过注释实现IOC"部分,我们指的是使用Spring的注解驱动配置来管理对象之间的依赖关系。例如,@Autowired注解可以自动将匹配类型的bean注入到需要它的类中,无需显式地创建对象实例。此外,@Component、@Service...

    Spring_in_Action

    1. **依赖注入**:Spring的核心特性之一,如何通过XML或注解方式实现对象间的依赖关系管理,以及如何使用@Autowired和@Qualifier注解。 2. **AOP(面向切面编程)**:解释什么是切面,如何定义和使用通知(advice)...

    spring源码下载

    Spring框架是Java开发中不可或缺的一部分,它以其模块化、易用性和灵活性著称。Spring Framework 2.5.6是该框架的一个较早版本,但仍然包含了许多...Spring源码的结构清晰,注释丰富,是学习Java企业级开发的宝贵资源。

    spring 4.0.7 源码

    在源码中,可以看到`org.springframework.beans`和`org.springframework.context`包下的类,如`BeanFactory`和`ApplicationContext`,它们是IoC容器的实现基础。 2. **AOP(Aspect Oriented Programming)**:...

    Spring动态代理二.docx

    升级后,靠注释自动装配@Autowired 实现,但是底层还是靠 ref 实现。 集成 web: 在 web.xml 文件内添加监听,用来加载 Spring 的主配置文件。标准配置文件路径:配置文件放入 config 文件夹(日志配置文件保留在...

    Struts2+Spring+Ibatis整合

    5. **Action类和Service层**:在Struts2中定义Action类,这些类通常是Spring中的Bean,通过@Autowired注解实现依赖注入。Service层负责业务逻辑,它调用DAO层进行数据操作。 6. **DAO层**:使用iBatis提供的...

    静态方法中调用Spring注入过程解析

    Spring框架使用IOC容器来管理Bean的生命周期,使得开发者可以专注于业务逻辑的实现。 @Autowired @Autowired是Spring框架提供的一个注解,用于实现依赖注入。@Autowired可以将Bean自动装配到容器中,使得开发者...

    spring-master-class:Spring Framework 5的更新介绍。成为了解Spring In Depth核心功能的专家。 您将在课程期间编写单元测试,AOP,JDBC和JPA代码。 包括对Spring Boot,JPA,Eclipse,Maven,JUnit和Mockito的介绍

    您将了解如何使用Spring注释-@ Autowired,@ Component,@ Service,@ Repository,@ Configuration,@ Primary...。 您将深入了解Spring MVC-DispatcherServlet,模型,控制器和ViewResolver 您将使用各种Sp

Global site tag (gtag.js) - Google Analytics