依赖注入
依赖注入(DI)是一种设计模式,主要用于将给定的属性注入到IOC容器创建的对象中(即对Bean的成员属性进行赋值)。
环境说明
- JDK 17
- Spring 6.0.6
实现依赖注入
Spring 实现依赖注入主要有两种方法:
- Setter方法依赖注入:通过成员属性对应的set方法完成注入。
- 构造方法依赖注入:通过构造方法完成注入。
Setter 依赖注入
此方法要求注入的属性必须有对应的 set 方法:
public class Student {
private Teacher teacher;
public void setTeacher(Teacher teacher) {
this.teacher = teacher;
}
public void hello(){
System.out.println("Hello World!");
teacher.teach(); // 调用注入属性相关方法
}
}
对于Teacher接口有两个具体的实现,注入时可以选取具体注入哪个对象,使得对于对象属性的操作更加灵活:
public interface Teacher {
void teach();
}
public class ArtTeacher implements Teacher {
@Override
public void teach() {
System.out.println("我是美术老师,我教你画画!");
}
}
public class MusicTeacher implements Teacher {
@Override
public void teach() {
System.out.println("我是音乐老师,我教你弹钢琴!");
}
}
注入时需要配置被注入的对象,使得注入时可以使用 IOC 容器创建好的对象,然后将目标注入对象的 bean 标签展开,通过property 标签注入属性:
<bean name="artTeacher" class="com.example.bean.ArtTeacher"/>
<bean name="musicTeacher" class="com.example.bean.MusicTeacher"/>
<bean class="com.example.bean.Student">
<property name="teacher" ref="artTeacher"/>
</bean>
<!-- name表示注入的属性名,ref表示要注入的对象 -->
主方法调用注入属性相关方法使用注入属性:
public static void main(String[] args){
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("spring.xml"); //这里写上刚刚的名字
Student student = context.getBean(Student.class);
student.hello();
}
依赖注入除了注入一个 Bean 以外,还可以直接注入一个具体的值:
public class Student {
private String name;
public void setName(String name) {
this.name = name;
}
...
<bean class="com.example.bean.Student">
<property name="name" value="NameString"/>
</bean>
构造方法依赖注入
IOC 容器默认只会调用无参构造方法,这里可以编写有参构造方法进行依赖注入:
public class Student {
private String name;
private Teacher teacher;
public Student(String name, Teacher teacher) {
this.name = name;
this.teacher = teacher;
}
<bean name="artTeacher" class="com.example.bean.ArtTeacher"/>
<bean name="musicTeacher" class="com.example.bean.MusicTeacher"/>
<bean class="com.example.bean.Student">
<constructor-arg name="name" value="xiaoming"/>
<constructor-arg name="teacher" ref="musicTeacher"/>
</bean>
如果存在多个构造方法,可以通过设置constructor-arg标签的 type 、name等属性来匹配目标构造方法。
除了注入 Bean 或者具体值,两种注入的方式对于List、Map等集合以及数组都有着特殊的支持。
自动装配
setter 方法依赖注入和构造方法依赖注入需要对 property 标签和 constructor-arg 标签参数进行配置,这时为了方便可以开启自动装配,即让 IOC 容器自己去寻找需要填入的值:
setter方法注入(添加 autowire 属性,有两个值,一个是byName,一个是byType):
<bean class="com.example.bean.Student" autowire="byType"/>
构造方法注入:
<bean class="com.example.bean.Student" autowire="constructor"/>
虽然自动装配方便快捷,但是过于机械,比如setter方法和构造方法注入无法注入具体值、存在多个同一类型的Bean时无法注入等问题。
生命周期
可以在配置文件中为 Bean 指定初始方法和销毁方法,以便在创建和销毁Bean时执行一些额外的任务:
public void init(){
System.out.println("我是对象初始化时要做的事情!");
}
public void destroy(){
System.out.println("我是对象销毁时要做的事情!");
}
<bean class="com.example.bean.Student" autowire="constructor" init-method="init" destroy-method="destroy"/>
创建和销毁Bean:
public static void main(String[] args){
//当容器创建时,默认情况下Bean都是单例的,对象构造完成后,会执行初始化方法
ClassPathXmlApplicationContext context = new ClassPathXmlApplicationContext("spring.xml");
Student student = context.getBean(Student.class);
student.hello();
// 调用close方法关闭容器,此时容器内存放的Bean也会被一起销毁
context.close();
}