spring中的依赖注入
Dependency Injection
IOC的作用:削减/降低程序间的依赖关系/耦合。依赖关系的管理都交给了spring来维护
依赖关系:在当前类中要使用的其他类的对象,由spring为我们提供,我们只需要在配置文件中说明
依赖关系的维护称之为:依赖注入
1. 能注入的数据有三类
- 基本类型和String
- 其他bean类型(在配置文件中或者注解配置过的bean)
- 复杂类型(集合类型)
2. 注入的方式有三种
- 使用构造函数提供
- 使用set方法提供
- 使用注解提供
1.1 使用构造函数提供
如果是经常变化的数据,并不适用于注入方式。
-
bean的标签:constructor-arg
标签出现的位置:bean标签的内容
标签中的属性:
- type:用于指定要注入的数据的数据类型,该数据类型也是构造函数中某个或某些数据类型
- index:用于指定要注入的数据给构造函数中指定索引位置的参数。索引的位置从0开始
- name:用于指定给构造函数中指定名称的参数赋值(常用)
以上三个用于指定给构造函数中的哪个参数赋值
- value:用于提供基本类型和String类型的数据
- ref:用于指定其他的bean类型数据。它指的是在Spring核心容器中出现过的bean对象,也就是在bean.xml文件中配置过的bean对象 -
优势:在获取bean对象时,注入必须的注入数据是必须的操作,否则对象无法创建成功
-
弊端:改变了bean对象的实例化方式,使我们在创建对象时,如果用不到这些数据也必须提供
- bean.xml
<!--构造函数注入-->
<bean id="accountService" class="com.itheima.service.impl.AccountServiceImpl" >
<constructor-arg name="name" value="test"></constructor-arg>
<constructor-arg name="age" value="18"></constructor-arg>
<constructor-arg name="birthday" ref="now"></constructor-arg>
</bean>
<bean id="now" class="java.util.Date"></bean>
-
将test注入给构造函数中是String类型的成员,如果构造函数中有两个String类型,就不可用,所以type并不能独立实现注入
-
index注入的类型要和索引位置的类型对应,否则无法注入成功
-
spring中xml文件都是String类型,但是对于age的Integer类型,是可以自动进行类型转换的
-
对于Date类型,Spring是无法进行转换的
-
在bean.xml中配置一个Date对象,使用ref引用这个对象
-
String类型注入还可以处理中文,不会乱码
- AccountServiceImpl.java
/**
* 账户的业务层实现类
*/
public class AccountServiceImpl implements IAccountService {
private String name;
private Integer age;
private Date birthday;
public AccountServiceImpl(String name, Integer age, Date birthday){
this.name = name;
this.age = age;
this.birthday =birthday;
}
public void saveAccount(){
System.out.println("service中的saveAccount執行了。" + name + "," + age + "," + birthday);
}
}
- Client.java
public class Client {
/**
* 获取spring的IOC核心容器,并根据id获取对象
* @param args
*/
public static void main(String[] args) {
//1、获取核心容器对象。配置文件在根目录,直接写文件名即可。
ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");
//2、根据id获取bean对象
IAccountService as = (IAccountService) ac.getBean("accountService2"); //使用注入的数据
as.saveAccount();
/*IAccountService as = new AccountServiceImpl2(); //使用默认构造函数
as.saveAccount();*/
((ClassPathXmlApplicationContext) ac).close();
}
}
1.2 使用set方法提供(更常用)
-
bean的标签:property
标签出现的位置:bean标签的内容
标签中的属性:
- name:用于指定注入时所调用的set方法名称,不管变量名是什么,看的是set方法的名称!
- value:用于提供基本类型和String类型的数据
- ref:用于指定其他的bean类型数据。它指的是在Spring核心容器中出现过的bean对象,也就是在bean.xml文件中配置过的bean对象 -
优势:创建对象时没有明确的限制,可以直接使用默认构造函数
-
弊端:如果某个成员必须有值,则获取对象时有可能set方法没有执行
-
bean.xml
<!--set方法注入-->
<bean id="accountService2" class="com.itheima.service.impl.AccountServiceImpl2" >
<property name="userName" value="TEST"></property>
<property name="age" value="21"></property>
<property name="birthday" ref="now"></property>
</bean>
- AccountServiceImpl2.java
/**
* 账户的业务层实现类
*/
public class AccountServiceImpl2 implements IAccountService {
private String name;
private Integer age;
private Date birthday;
public void setUserName(String name) {
this.name = name;
}
public void setAge(Integer age) {
this.age = age;
}
public void setBirthday(Date birthday) {
this.birthday = birthday;
}
public void saveAccount(){
System.out.println("service中的saveAccount執行了。" + name + "," + age + "," + birthday);
}
}
1.2.1 复杂类型的注入
<!--集合类型/复杂类型的注入-->
<bean id="accountService3" class="com.itheima.service.impl.AccountServiceImpl3" >
<property name="myStrs">
<array>
<value>AAA</value>
<value>BBB</value>
<value>CCC</value>
</array>
</property>
<property name="myList">
<list>
<value>AAA</value>
<value>BBB</value>
<value>CCC</value>
</list>
</property>
<property name="mySet">
<set>
<value>AAA</value>
<value>BBB</value>
<value>CCC</value>
</set>
</property>
<property name="myMap">
<map>
<entry key="testA" value="aaa"></entry>
<entry key="testB">
<value>BBB</value>
</entry>
</map>
</property>
<property name="myPropers">
<props>
<prop key="testC">ccc</prop>
<prop key="testD">ddd</prop>
</props>
</property>
</bean>
- AccountServiceImpl.java
/**
* 账户的业务层实现类
*/
public class AccountServiceImpl3 implements IAccountService {
private String[] myStrs;
private List<String> myList;
private Set<String> mySet;
private Map<String, String> myMap;
private Properties myPropers;
public void setMyStrs(String[] myStrs) {
this.myStrs = myStrs;
}
public void setMyList(List<String> myList) {
this.myList = myList;
}
public void setMySet(Set<String> mySet) {
this.mySet = mySet;
}
public void setMyMap(Map<String, String> myMap) {
this.myMap = myMap;
}
public void setMyPropers(Properties myPropers) {
this.myPropers = myPropers;
}
public void saveAccount(){
System.out.println(Arrays.toString(myStrs));
System.out.println(myList);
System.out.println(mySet);
System.out.println(myMap);
System.out.println(myPropers);
}
}
- Client.java
public class Client {
/**
* 获取spring的IOC核心容器,并根据id获取对象
* @param args
*/
public static void main(String[] args) {
//1、获取核心容器对象。配置文件在根目录,直接写文件名即可。
ApplicationContext ac = new ClassPathXmlApplicationContext("bean.xml");
//2、根据id获取bean对象
IAccountService as = (IAccountService) ac.getBean("accountService3");
as.saveAccount();
((ClassPathXmlApplicationContext) ac).close();
}
}
总结
使用Spring中的IOC解决程序耦合问题
Spring对Bean的管理细节
Spring中的依赖注入(一)
1、Spring的IOC可以解决什么问题
2、如何搭建出基于xml的开发环境
3、如何使用Spring注入实现类之间的依赖关系
4、通过IOC改造代码