依赖注入
依赖注入(Dependency Injection,简称DI)是Spring对IOC的一种实现方式。
通过控制反转,对象在被创建的时候,由Spring容器将其所依赖的对象的引用传递给它,也可以理解为,依赖被注入到对象中,所以就被称作为依赖注入。
就是通过Spring,将所有的对象放在一个容器中,再由这个容器将其中被依赖的对象注入到需要该依赖的对象中,听起来确实有点绕,我们看下面的图:
其中对象的创建和管理,都交由Spring容器来管理,它在创建对象时,会将该对象依赖的对象注入到该对象中。
我们来看看Spring的具体实现方式:
博主的演示环境:
- idea编辑器
- Maven项目
1. 先导入Spring相关依赖
建议使用Maven直接导入,这里博主就不再提供各个jar包的下载地址了,需要的可以直接去maven仓库搜索下载,博主这里直接导入依赖即可:
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-webmvc</artifactId>
<version>5.2.4.RELEASE</version>
</dependency>
因为Maven会自动导入该包依赖的其他所有的包,所有博主推荐在使用Spring时直接导入该依赖。
2. 准备Spring的配置文件
我们在resources目录下创建一个beans.xml
的配置文件,名字可以任意取,Spring官方建议使用applicationContext.xml
,我们在该配置文件中引入头约束:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xsi:schemaLocation="http://www.springframework.org/schema/beans
https://www.springframework.org/schema/beans/spring-beans.xsd">
</beans>
我们简单了解一下该配置文件中的一些相关配置(这里仅仅介绍beans约束下的一些配置):
- bean:配置一个bean对象,交由Spring接管
- id:bean的唯一标识符,可以理解为我们new变量时的变量名
- class:bean对象所对应的全限定类名
- name:也是别名 而且该属性还能取多个别名
- scope:作用域(这里了解singleton单例和prototype原型即可)
- autowire:自动注入方式(byName和byType)
- alias:别名,就是为一个bean对象可以取多个别名
- name:指定bean的唯一标识符id
- alias:别名
- import:用于将多个xml文件导入一个xml文件中,多用于协同开发
- resource:指定导入的资源
3. 准备三个Java实体类
Address:
package com.ara.pojo;
public class Address {
private String name;
public Address(String name) {
this.name = name;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
@Override
public String toString() {
return "Address{" +
"name='" + name + '\'' +
'}';
}
}
Student:
package com.ara.pojo;
import java.util.*;
public class Student {
private String name;
private Address address;
private String[] books;
private List<String> hobbies;
private Map<String,String> card;
private Set<String> games;
private Properties info;
private String wife;
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public Address getAddress() {
return address;
}
public void setAddress(Address address) {
this.address = address;
}
public String[] getBooks() {
return books;
}
public void setBooks(String[] books) {
this.books = books;
}
public List<String> getHobbies() {
return hobbies;
}
public void setHobbies(List<String> hobbies) {
this.hobbies = hobbies;
}
public Map<String, String> getCard() {
return card;
}
public void setCard(Map<String, String> card) {
this.card = card;
}
public Set<String> getGames() {
return games;
}
public void setGames(Set<String> games) {
this.games = games;
}
public Properties getInfo() {
return info;
}
public void setInfo(Properties info) {
this.info = info;
}
public String getWife() {
return wife;
}
public void setWife(String wife) {
this.wife = wife;
}
@Override
public String toString() {
return "Student{" +
"name='" + name + '\'' +
", address=" + address +
", books=" + Arrays.toString(books) +
", hobbies=" + hobbies +
", card=" + card +
", games=" + games +
", info=" + info +
", wife='" + wife + '\'' +
'}';
}
}
User:
package com.ara.pojo;
public class User {
private String name;
private int age;
public User() {
}
public User(String name, int age) {
this.name = name;
this.age = age;
}
public String getName() {
return name;
}
public void setName(String name) {
this.name = name;
}
public int getAge() {
return age;
}
public void setAge(int age) {
this.age = age;
}
@Override
public String toString() {
return "User{" +
"name='" + name + '\'' +
", age=" + age +
'}';
}
}
博主在Address中提供了带参构造和对应属性的set方法,在Student类中准备了各种各样的属性值,并提供了对应属性的set方法,在User类中也准备了无参构造和带参构造,以及对应属性的set方法,便于下面演示注入。
4. 配置bean到Spring容器中
这里只需要在第二步创建的配置文件中添加即可
4.1 构造器注入
这里我们以Address为演示(这里博主为该类值提供了一个构造方法,所以创建bean时,必须为它的构造方法赋值,Spring会根据配置的bean去找对应的构造方法):
- 按构造方法中参数下标注入
<bean id="address1" class="com.ara.pojo.Address">
<!--为构造方法中下标为0的参数赋值为China-->
<constructor-arg index="0" value="China"/>
</bean>
- 按构造方法中参数类型注入(不建议使用,如果几个参数类型一致,后果。。。)
<bean id="address2" class="com.ara.pojo.Address">
<!--为构造方法中类型为java.lang.String的参数赋值为四川-->
<constructor-arg type="java.lang.String" value="四川"/>
</bean>
- 按构造方法重参数名字注入
<bean id="address3" class="com.ara.pojo.Address">
<!--为构造方法中名为name的参数赋值为成都-->
<constructor-arg name="name" value="成都"/>
</bean>
4.2 set注入
这里以Student为例(注意一定要存在对应属性的set方法哦):
<bean id="student" class="com.ara.pojo.Student">
<!-- 1.普通简单值注入 -->
<property name="name" value="Ara_Hu"/>
<!-- 2.bean引用注入 ref -->
<property name="address" ref="address1"/>
<!-- 3.数组注入 内嵌array标签 -->
<property name="books">
<array>
<value>西游记</value>
<value>红楼梦</value>
<value>水浒传</value>
<value>三国演义</value>
</array>
</property>
<!-- 4.list注入 -->
<property name="hobbies">
<list>
<value>听歌</value>
<value>打代码</value>
<value>抽烟</value>
<value>喝酒</value>
</list>
</property>
<!-- 5.map注入 -->
<property name="card">
<map>
<entry key="身份证" value="000000000000" />
<entry key="银行卡" value="111111111111" />
</map>
</property>
<!-- 6.set注入 -->
<property name="games">
<set>
<value>LOL</value>
<value>CF</value>
<value>QQ</value>
<value>CS</value>
</set>
</property>
<!-- 7.properties注入 -->
<property name="info">
<props>
<prop key="学号">20111111</prop>
<prop key="年级">2011</prop>
<prop key="专业">计算机</prop>
</props>
</property>
<!-- 8.null注入 -->
<property name="wife">
<null/>
</property>
</bean>
4.3 扩展注入方式
p命名空间和c命名空间 这里需要注意需要在配置文件中导入头约束:
<?xml version="1.0" encoding="UTF-8"?>
<beans xmlns="http://www.springframework.org/schema/beans"
xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
xmlns:p="http://www.springframework.org/schema/p"
xmlns:c="http://www.springframework.org/schema/c"
xsi:schemaLocation="http://www.springframework.org/schema/beans
https://www.springframework.org/schema/beans/spring-beans.xsd">
</beans>
这里就以User类为例
- P命名空间
<!-- p命名空间就相当于<property>标签的简化版 -->
<bean id="user1" class="com.ara.pojo.User" p:name="Ara_Hu" p:age="18"/>
- C命名空间
<!-- c命名空间就相当于<constructor-arg>标签的简化版 -->
<bean id="user2" class="com.ara.pojo.User" c:name="Ara_Hu" c:age="20"/>
在这几个例子中,博主将几乎所有的注入类型都演示了,接下来我们测试一下
5. 测试代码
@Test
public void springTest01(){
//读取配置文件 获取Spring上下文对象 可以理解为Spring的容器对象
ApplicationContext context = new ClassPathXmlApplicationContext("beans.xml");
//获取学生对象
Student student = context.getBean("student",Student.class);
System.out.println(student);
//获取User对象
User user = context.getBean("user1", User.class);
System.out.println(user);
}
测试结果:
说明注入成功了。
通过上述演示,我们从头至尾都没有手动的new过对象,而是从Spring的上下文对象中获取出的对象。