zoukankan      html  css  js  c++  java
  • springMVC学习(注解实现依赖注入)

    原文:http://blog.csdn.net/mockingbirds/article/details/45399691

    上一篇博客,学习了spring的依赖注入,即利用spring容器来为类中的属性赋值,分为两种赋值方法,利用set和利用构造方法,我们都知道,如果我需要为某一个属性赋值的话,必须为该属性写上set方法,那么大家有没有想过一个问题,如果我们一个类中有很多个属性,我们会生成大量的set方法,如果用构造方法来赋值,即<constructor-arg index="" type="" ref="" value=""></constructor-arg>,这样也会存在很多个这样的标签,因为一个<constructor-arg index="" type="" ref="" value=""></constructor-arg>只能为一个属性赋值,为了解决这些难题,我们可以利用注解来为属性进行赋值。

           同样,我们新建两个类ClassInfo.Java和Teacher.java

    [html] view plain copy
     
    1. package com.test.spring.di;  
    2.   
    3. public class ClassInfo {  
    4.       
    5.     public void printClassInfo() {  
    6.         System.out.println("this is classInfo....");  
    7.     }  
    8. }  
    [html] view plain copy
     
    1. package com.test.spring.di;  
    2.   
    3. import javax.annotation.Resource;  
    4.   
    5. public class Teacher {  
    6.       
    7.     private String teacherName;  
    8.     @Resource  
    9.     private ClassInfo classInfo;  
    10.       
    11.     public void printClassInfo() {  
    12.         this.classInfo.printClassInfo();  
    13.     }  
    14.       
    15. }  

          注意,这里我们给需要被赋值的属性上添加了@Resource的注解。其中包含了基本的String类型以及ClassInfo的引用类型。接下来我们来写一个applicationContext.xml文件:

    注意:因为我们使用注解来为属性赋值的,所以我们需要导入命名空间,如下:

    xmlns:context="http://www.springframework.org/schema/context"

    http://www.springframework.org/schema/context
    http://www.springframework.org/schema/context/spring-context-2.5.xsd"

         其次需要导入依赖注入的注解解析器:

    <context:annotation-config></context:annotation-config>

    3.导入需要被spring容器来管理的类。

           经过上面散步操作之后呢,applicationContext.xml是这个样子:

    [html] view plain copy
     
    1. <?xml version="1.0" encoding="UTF-8"?>  
    2. <beans xmlns="http://www.springframework.org/schema/beans"  
    3. xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance" xmlns:context="http://www.springframework.org/schema/context"  
    4.     xsi:schemaLocation="http://www.springframework.org/schema/beans  
    5.            http://www.springframework.org/schema/beans/spring-beans-2.5.xsd  
    6.            http://www.springframework.org/schema/context  
    7.            http://www.springframework.org/schema/context/spring-context-2.5.xsd  
    8.            ">  
    9.     <context:annotation-config></context:annotation-config>  
    10.     <bean id="classInfo" class="com.test.spring.di.ClassInfo"></bean>  
    11.     <bean id="teacher" class="com.test.spring.di.Teacher"></bean>  
    12. </beans>  

    此时,编写我们的测试类TestAnotation.java

    [java] view plain copy
     
    1. ApplicationContext applicationContext = new ClassPathXmlApplicationContext("com/test/spring/di/applicationContext.xml");  
    2. Teacher teacher = (Teacher) applicationContext.getBean("teacher");  
    3. teacher.printClassInfo();  

    此时打印如下:

    this is classInfo....

    这充分说明了利用注解为Teacher类中的classInfo对象赋值成功了。在这里需要注意:如果我么你的@Resource的注解的值为"",那么spring将会将容器中id为该属性的类注入给该属性。什么意思呢??我举个栗子:

    我将上面的@Resource的值修改为这样:@Resource(name="test")此时由于name的值不是"",所以spring容器将不会试图匹配与该属性相同的id对应的类,而是将id为test的类注入给该属性,然而,此时并没有id为test的类,所以会抛出如下异常:

    No bean named 'test' is defined

    此时我将classInfo对应的id改为test,再次运行发现程序是ok的。

          其实spring有自己定义的注解的,细心的程序员可能发现了@Resource是java官方提供的一个注解,并不是spring的,我们可以将@Resource替换成@Autowired,这个@Autowired是根据类型进行匹配的。那我如果非要按照id来进行匹配怎么办呢?别急,spring为我们提供了一个@Qualifier,我们如果要匹配spring容器中id为classInfo的类,可以这样写:@Qualifier("classInfo"),注意:这里还需要加上@Autowired,如下:

    @Autowired
    @Qualifier("classInfo")
    private ClassInfo classInfo;
            大家有没有想过,既然spring是用来管理bean的,难道就没有生命周期的管理???是有的。我们为ClassInfo类添加如下初始化和销毁方法:

    [html] view plain copy
     
    1. @PostConstruct  
    2. public void init() {  
    3.     System.out.println("classInfo init....");  
    4. }  
    5. @PreDestroy  
    6. public void destroy() {  
    7.     System.out.println("classInfo destroy.....");  
    8. }  

    顾名思义,@PostConstruct表示紧接着在构造方法之后执行的,@PreDestroy表示在销毁前执行的,然后编写测试代码如下:

    [html] view plain copy
     
    1. ApplicationContext applicationContext = new ClassPathXmlApplicationContext("com/test/spring/di/applicationContext.xml");  
    2. ClassPathXmlApplicationContext classContext = (ClassPathXmlApplicationContext) applicationContext;  
    3. Teacher teacher = (Teacher) classContext.getBean("teacher");  
    4. teacher.printClassInfo();  
    5. classContext.close();  

    此时运行工程,会打印如下信息:

    classInfo init....
    this is classInfo....
    classInfo destroy.....
    我们发现确实执行了初始化和销毁的方法,说明利用注解来实现声明周期的管理也是可以的。

          但是,注意spring的注解只能用于引用类型。 

          那我能不能在applicationContext.xml中对于bean的声明也不写呢??是可以的,怎么做的,同样分为以下三步:

    1.导入命名空间:

    xmlns:context="http://www.springframework.org/schema/context"

    http://www.springframework.org/schema/context
    http://www.springframework.org/schema/context/spring-context-2.5.xsd"

    2.启动扫描类的注解解析器

    <context:annotation-config></context:annotation-config>
    3.启动依赖注入的注解解析器

    <context:component-scan base-package="com.test.spring.di"></context:component-scan>

    这里是扫描"com.test.spring.di"包和该包下的所有的类。

    4.在需要被spring来管理的类上加上@Component的注解

    注意:和@Resource比较相似,@Component是默认匹配类名的第一个字母小写的,比如我在Teacher类上加了@Component这个注解,那么可以这样得到该类对象:

    ApplicationContext applicationContext = new ClassPathXmlApplicationContext("com/test/spring/di/applicationContext.xml");
    Teacher teacher =  (Teacher) applicationContext.getBean("teacher");

    如果Teacher类上写了这样一个@Component("teacherId"),那么此时就需要通过

    Teacher teacher =  (Teacher) applicationContext.getBean("teacherId");

    才可以得到Teacher对象。这里的@Component是一个比较宽泛的泛型,spring为我们提供了更详细的注解配置:

    @Controller  配置控制器的,控制视图的跳转  
    @Service     配置service的,不是Android中的service

    @Repository    配置dao的,控制操作数据库

         我们发现这三个注解是对web开发中的MVC编程很好的一个支持。

          恩,今天spring注解,就到这里吧,希望大家看了都能理解。

  • 相关阅读:
    分析一个文本文件(英文文章)中各个词出现的频率,并且把频率最高的10个词打印出来
    求一个数组中的最大整数
    一个统计文本文件中各个英文单词出现频率的问题,并且输出频率最高的10个词
    Python学习一:基础语法
    Spring学习之二
    Spring学习之装配Bean
    Spring学习一
    缓存之ehcache
    解决axios传递参数后台无法接收问题
    服务端解决跨域问题
  • 原文地址:https://www.cnblogs.com/silentdoer/p/7171604.html
Copyright © 2011-2022 走看看