Inverse of Control 反转控制,把创建对象的权利交给框架,包括依赖注入和依赖查找。
以前对象的创建时由我们开发人员自己维护,包括依赖关系也是自己注入。
使用了Spring之后,对象的创建及依赖关系可以由Spring完成创建以及注入。
反转控制就是反转了对象的创建方式,从我们自己创建反转给了程序(Spring)。
//User对象
import java.util.*;
public class User {
private Integer id;
private String username;
private Character sex;
private Date birthday;
public User() {
}
public User(Integer id, String username, Character sex, Date birthday) {
this.id = id;
this.username = username;
this.sex = sex;
this.birthday = birthday;
}
//getter and setter此处省略
}
1. 引入依赖
<dependency>
<groupId>org.springframeworkgroupId>
<artifactId>spring-contextartifactId>
<version>5.0.2.RELEASEversion>
dependency>
2.创建UserDao接口和UserDaoImpl实现类。
3. 创建beans.xml配置文件
<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
http://www.springframework.org/schema/beans/spring-beans.xsd">
<bean id="userDao" class="com.testfan.dao.impl.UserDaoImpl">bean>
beans>
4. 创建容器对象,根据id获取对象
//创建spring的IOC容器
ApplicationContext ac = new ClassPathXmlApplicationContext("beans.xml");
//获取对象
Object userDao = ac.getBean("userDao");
System.out.println(userDao);
作用:用于配置对象让 spring 来创建的。默认情况下它调用的是类中的无参构造函数。如果没有无参构造函数则不能创建成功。
属性:
单例对象: scope="singleton"
一个应用只有一个对象的实例。它的作用范围就是整个引用。
生命周期:
对象出生:当应用加载,创建容器时,对象就被创建了。
对象活着:只要容器在,对象一直活着。
对象死亡:当应用卸载,销毁容器时,对象就被销毁了。
多例对象: scope="prototype"
每次访问对象时,都会重新创建对象实例。
生命周期:
对象出生:当使用对象时,创建新的对象实例。
对象活着:只要对象在使用中,就一直活着。
对象死亡:当对象长时间不用时,被 java 的垃圾回收器回收了。
第一种方式:使用默认无参构造函数
<bean id="accountService" class="com.testfan.service.impl.AccountServiceImpl"/>
第二种方式: spring 管理静态工厂-使用静态工厂的方法创建对象
/**
* 模拟一个静态工厂,创建业务层实现类
*/
public class StaticFactory {
public static IAccountService createAccountService(){
return new AccountServiceImpl();
}
}
<bean id="accountService" class="com.testfan.factory.StaticFactory"
factory-method="createAccountService">bean>
第三种方式: spring 管理实例工厂-使用实例工厂的方法创建对象
/**
* 模拟一个实例工厂,创建业务层实现类
* 此工厂创建对象,必须现有工厂实例对象,再调用方法
*/
public class InstanceFactory {
public IAccountService createAccountService(){
return new AccountServiceImpl();
}
}
<bean id="instancFactory" class="com.testfan.factory.InstanceFactory">bean>
<bean id="accountService" factory-bean="instancFactory"
factory-method="createAccountService">bean>
Dependency Injection依赖注入,实现IoC思想需要DI做支持。 它是 spring 框架核心 IoC的具体实现。
我们的程序在编写时, 通过控制反转,把对象的创建交给了 spring,但是代码中不可能出现没有依赖的情况。IoC解耦只是降低他们的依赖关系,但不会消除。 例如:我们的业务层仍会调用持久层的方法。那这种业务层和持久层的依赖关系, 在使用 spring 之后, 就让 spring 来维护了。简单的说,就是坐等框架把持久层对象传入业务层,而不用我们自己去获取。
<bean id="birthday" class="java.util.Date">bean>
<bean id="user" class="com.testfan.domain.User">
<property name="username" value="张三">property>
<property name="birthday" ref="birthday">property>
<property name="sex" value="男">property>
<property name="id" value="5">property>
bean>
<bean id="user" class="com.testfan.domain.User">
<constructor-arg name="id" value="3">constructor-arg>
<constructor-arg name="username" value="王五">constructor-arg>
<constructor-arg name="sex" value="男">constructor-arg>
<constructor-arg name="birthday" ref="birthday">constructor-arg>
bean>
<bean id="birthday" class="java.util.Date">bean>
import java.util.*;
public class User {
private List<String> list;
private Set<String > set;
private String[] strs;
private Map<String ,String> map;
private Properties properties;
public List<String> getList() {
return list;
}
public void setList(List<String> list) {
this.list = list;
}
public Set<String> getSet() {
return set;
}
public void setSet(Set<String> set) {
this.set = set;
}
public String[] getStrs() {
return strs;
}
public void setStrs(String[] strs) {
this.strs = strs;
}
public Map<String, String> getMap() {
return map;
}
public void setMap(Map<String, String> map) {
this.map = map;
}
public Properties getProperties() {
return properties;
}
public void setProperties(Properties properties) {
this.properties = properties;
}
}
<bean id="user4" class="com.testfan.domain.User">
<property name="list">
<set>
<value>javaSEvalue>
<value>javaEEvalue>
<value>javaMEvalue>
set>
property>
<property name="set">
<array>
<value>javaSEvalue>
<value>javaEEvalue>
<value>javaMEvalue>
array>
property>
<property name="strs">
<list>
<value>javaSEvalue>
<value>javaEEvalue>
<value>javaMEvalue>
list>
property>
<property name="map">
<props>
<prop key="五" >fiveprop>
<prop key="六" >sixprop>
<prop key="七" >sevenprop>
props>
property>
<property name="properties">
<map>
<entry key="一" value="one">entry>
<entry key="二" value="two">entry>
<entry key="三" value="three">entry>
<entry key="四" value="four">entry>
map>
property>
bean>
<bean id="birthday" class="java.util.Date">bean>
<bean id="user" class="com.testfan.domain.User">
<property name="username" value="张三">property>
<property name="birthday" ref="birthday">property>
<property name="sex" value="男">property>
<property name="id" value="5">property>
bean>
Spring中的工厂类
BeanFactory接口是 Spring 容器中的顶层接口,针对原始接口的实现类功能较为单一。
BeanFactory接口实现类的容器,特点是每次在获得对象时才会创建对象,即什么时候使用什么时候创建
这主要是因为,BeanFactory是早年使用的接口,当时硬件资源比较匮乏,内存配置低,所以当时设计的框架资源使用比较节约。
ApplicationContext 是BeanFactory接口的子接口。
ApplicationContext接口实现类的容器,特点是每次容器启动时就会创建容器中配置的所有对象,并提供更多功能
典型的实现类
创建对象的时间点不一样。
ApplicationContext:只要一读取配置文件, 默认情况下就会创建对象。
BeanFactory:什么使用什么时候创建对象。
结论:web开发中,使用applicationContext. 在资源匮乏的环境可以使用BeanFactory.