Silentdoer

导航

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注解,就到这里吧,希望大家看了都能理解。

posted on 2017-07-14 17:35  Silentdoer  阅读(739)  评论(0编辑  收藏  举报