上一篇博客,学习了spring的依赖注入,即利用spring容器来为类中的属性赋值,分为两种赋值方法,利用set和利用构造方法,我们都知道,如果我需要为某一个属性赋值的话,必须为该属性写上set方法,那么大家有没有想过一个问题,如果我们一个类中有很多个属性,我们会生成大量的set方法,如果用构造方法来赋值,即
同样,我们新建两个类ClassInfo.java和Teacher.java
package com.test.spring.di;
public class ClassInfo {
public void printClassInfo() {
System.out.println("this is classInfo....");
}
}
package com.test.spring.di;
import javax.annotation.Resource;
public class Teacher {
private String teacherName;
@Resource
private ClassInfo classInfo;
public void printClassInfo() {
this.classInfo.printClassInfo();
}
}
注意,这里我们给需要被赋值的属性上添加了@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"
其次需要导入依赖注入的注解解析器:
3.导入需要被spring容器来管理的类。
经过上面散步操作之后呢,applicationContext.xml是这个样子:
此时,编写我们的测试类TestAnotation.java
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("com/test/spring/di/applicationContext.xml");
Teacher teacher = (Teacher) applicationContext.getBean("teacher");
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类添加如下初始化和销毁方法:
@PostConstruct
public void init() {
System.out.println("classInfo init....");
}
@PreDestroy
public void destroy() {
System.out.println("classInfo destroy.....");
}
顾名思义,@PostConstruct表示紧接着在构造方法之后执行的,@PreDestroy表示在销毁前执行的,然后编写测试代码如下:
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("com/test/spring/di/applicationContext.xml");
ClassPathXmlApplicationContext classContext = (ClassPathXmlApplicationContext) applicationContext;
Teacher teacher = (Teacher) classContext.getBean("teacher");
teacher.printClassInfo();
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"
3.启动依赖注入的注解解析器
这里是扫描"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注解,就到这里吧,希望大家看了都能理解。
源码下载