1. IOC和DI的基本概念

1.1 IOC(Inversion of Control 控制反转):

  1. 其思想是反转资源获取的方向。传统的资源查找方式要求组件向容器发起请求查找资源,作为回应,容器适时的返回资源。

  2. 而应用了 IOC 之后,则是容器主动地将资源推送给它所管理的组件,组件所要做的仅是选择一种合适的方式来接受资源。

  3. 这种行为也被称为查找的被动形式。这体现的是一种设计思想,而不是具体的实现。

1.2 DI(Dependency Injection 依赖注入) — IOC 的另一种表达方式

  1. 组件以一些预先定义好的方式(例如: setter 方法)接受来自于容器的资源注入。

  2. DI是实现了IOC设计思想的具体行为,spring容器就是通过DI来实现了IOC的设计思想。

  3. 依赖注入的目的并不是给应用程序带来更多的功能,而是为了提升组件的重用频率,并为系统搭建一个灵活、可扩展的平台。通过依赖注入机制,我们只要通过简单的配置就可指定目标需要的资源,完成自身的业务逻辑,而不需要关心具体的资源来自何处,由谁实现。

2. Spring配置文件

2.1 Bean标签基本配置

用于配置对象交由Spring 来创建

默认情况下它调用的是类中的无参构造函数,如果没有无参构造函数则不能创建成功。

基本属性: id:Bean实例在Spring容器中的唯一标识 class:Bean的全限定名称

2.2 Bean标签范围配置

scope:指对象的作用范围,取值如下:

取值范围 说明
singleton 默认的,单列的
prototype 多例的
request WEB 项目中,Spring 创建一个 Bean 的对象,将对象存入到 request 域中
session WEB 项目中,Spring 创建一个 Bean 的对象,将对象存入到 session 域中
global session WEB 项目中,应用在 Portlet 环境,如果没有 Portlet 环境那么globalSession 相当 于 session

Bean的实例化个数:1个

Bean的实例化时机:当Spring核心文件被加载时,实例化配置的Bean实例

Bean的生命周期:

  • 对象创建:当应用加载,创建容器时,对象就被创建了

  • 对象运行:只要容器在,对象一直活着

  • 对象销毁:当应用卸载,销毁容器时,对象就被销毁了

示例

1
<bean id="userDao" class="cn.jyw.impl.UserDaoImpl" scope="singleton"></bean>
1
2
3
4
5
6
7
8
9
@Test
public void test1(){
ApplicationContext applicationContext = new
ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userDao1 = (UserDao) applicationContext.getBean("userDao");
UserDao userDao2 = (UserDao) applicationContext.getBean("userDao");
System.out.println(userDao1);
System.out.println(userDao2);
}

结果

1
2
3
UserDaoImpl对象创建
cn.jyw.impl.UserDaoImp@4abdb505
cn.jyw.impl.UserDaoImp@4abdb505

Bean的实例化个数:多个

Bean的实例化时机:当调用getBean()方法时实例化Bean

Bean的生命周期:

  • 对象创建:当使用对象时,创建新的对象实例

  • 对象运行:只要对象在使用中,就一直活着

  • 对象销毁:当对象长时间不用时,被 Java 的垃圾回收器回收了

示例

1
<bean id="userDao" class="cn.jyw.impl.UserDaoImpl" scope="prototype"></bean>
1
2
3
4
5
6
7
8
9
@Test
public void test1(){
ApplicationContext applicationContext = new
ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userDao1 = (UserDao) applicationContext.getBean("userDao");
UserDao userDao2 = (UserDao) applicationContext.getBean("userDao");
System.out.println(userDao1);
System.out.println(userDao2);
}

结果

1
2
3
4
UserDaoImpl对象创建
UserDaoImpl对象创建
cn.jyw.impl.UserDaoImpl@587d1d39
cn.jyw.impl.UserDaoImpl@58c1670b

2.3 Bean生命周期配置

1
<bean id="userDao" class="cn.jyw.impl.UserDaoImpl" init-method="init" destroy-method="destroy"></bean>
  • init-method:指定类中的初始化方法名称

    创建对象后 执行init方法

  • destroy-method:指定类中销毁方法名称

    销毁之前执行 destroy方法

ClassPathXmlApplicationContext.close 销毁执行

2.4 Bean实例化三种方式

它会根据默认无参构造方法来创建类对象,如果bean中没有默认无参构造函数,将会创建失败

1
<bean id="userDao" class="cn.jyw.impl.UserDaoImpl"></bean>

工厂的静态方法返回Bean实例

1
2
3
4
5
public class StaticFactory {
public static UserDao getUserDao(){
return new UserDaoImp();
}
}
1
<bean id="userDao" class="cn.jyw.factory.StaticFactory" factory-method="getUserDao"/>
1
2
3
4
5
6
7
@Test
public void test1(){
ApplicationContext applicationContext = new
ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userDao1 = (UserDao) applicationContext.getBean("userDao");
System.out.println(userDao1);
}
1
2
UserDaoImpl对象创建
cn.jyw.impl.UserDaoImpl@13c27452

由于工厂是不是静态方法 无法在创建时就使用方法 所以在使用工厂方法前需要先创建工厂

1
2
3
4
5
public class StaticFactory {
public UserDao getUserDao(){
return new UserDaoImpl();
}
}
1
2
<bean id="factory" class="cn.jyw.factory.StaticFactory"/>
<bean id="userDao" factory-bean="factory" factory-method="getUserDao"/>
1
2
3
4
5
6
7
@Test
public void test1(){
ApplicationContext applicationContext = new
ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userDao1 = (UserDao) applicationContext.getBean("userDao");
System.out.println(userDao1);
}
1
2
UserDaoImpl对象创建
cn.jyw.impl.UserDaoImpl@13c27452

3. Bean的依赖注入(DI)入门

  1. 创建UserService,UserService 内部在调用 UserDao的save() 方法
1
2
3
4
5
6
7
8
9
public class UserServiceImpl implements UserService {
@Override
public void save() {
ApplicationContext applicationContext = new
ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userDao = (UserDao)applicationContext.getBean("userDao");
userDao.save();
}
}
  1. 将 UserServiceImpl 的创建权交给 Spring
1
<bean id="userService" class="cn.jyw.service.impl.UserServiceImpl"/>
  1. 从 Spring 容器中获得 UserService 进行操作
1
2
3
4
ApplicationContext applicationContext = new 
ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService = (UserService)applicationContext.getBean("userService");
userService.save();

3.1 Bean的依赖注入分析

​ 目前UserService实例和UserDao实例都存在与Spring容器中,当前的做法是在容器外部获得UserService 实例和UserDao实例,然后在程序中进行结合。

Bean的依赖注入分析

​ 因为UserService和UserDao都在Spring容器中,而最终程序直接使用的是UserService,所以可以在 Spring容器中,将UserDao设置到UserService内部。

Bean的依赖注入原理

3.2 Bean的依赖注入概念

依赖注入(Dependency Injection):它是 Spring 框架核心 IOC 的具体实现。

在编写程序时,通过控制反转,把对象的创建交给了 Spring,但是代码中不可能出现没有依赖的情况。

IOC 解耦只是降低他们的依赖关系,但不会消除。

例如:业务层仍会调用持久层的方法。

那这种业务层和持久层的依赖关系,在使用 Spring 之后,就让 Spring 来维护了。

简单的说,就是坐等框架把持久层对象传入业务层,而不用我们自己去获取

3.3 Bean的依赖注入方式

  • 构造方法
  • set方法

属性注入是最常用的注入方式,通过 setter方法注入Bean 的属性值或依赖的对象。属性注入使用 <property> 元素, 使用 name 属性指定 Bean 的属性名称,value 属性或<value>子节点指定属性值

在UserServiceImpl中添加setUserDao方法

1
2
3
4
5
6
7
8
9
10
public class UserServiceImpl implements UserService {
private UserDao userDao;
public void setUserDao(UserDao userDao) {
this.userDao = userDao;
}
@Override
public void save() {
userDao.save();
}
}

配置Spring容器调用set方法进行注入

1
2
3
4
<bean id="userDao" class="cn.jyw.dao.impl.UserDaoImpl"/>
<bean id="userService" class="cn.jyw.service.impl.UserServiceImpl">
<property name="userDao" ref="userDao"/>
</bean>

P命名空间注入本质也是set方法注入,但比起上述的set方法注入更加方便,主要体现在配置文件中,如下: 首先,需要引入P命名空间:

1
2
xmlns:p="http://www.springframework.org/schema/p"
<bean id="userService" class="cn.jyw.service.impl.UserServiceImpl" p:userDao-ref="userDao"/>

通过构造方法注入Bean 的属性值或依赖的对象,它保证了 Bean 实例在实例化后就可以使用。构造器注入在 <constructor-arg> 元素里声明属性, <constructor-arg> 中没有 name 属性,可以按索引匹配入参,也可以按类型匹配入参

创建有参构造

1
2
3
4
5
6
7
8
9
10
public class UserServiceImpl implements UserService {
private UserDao userDao;
pub1ic UserServiceImpl(UserDao userDao) {
this. userDao = userDao;
}
@Override
public void save() {
userDao.save();
}
}

配置Spring容器调用有参构造时进行注入

1
2
3
4
<bean id="userDao" class="cn.jyw.dao.impl.UserDaoImpl"/>
<bean id="userService" class="cn.jyw.service.impl.UserServiceImpl">
<constructor-arg name="userDao" ref="userDao"/>
</bean>

3.4 Bean的依赖注入的数据类型

可用字符串表示的值,可以通过 元素标签或 value 属性进行注入

基本数据类型及其封装类、String 等类型都可以采取字面值注入的方式

若字面值中包含特殊字符(会使用到xml文件中的一些专用字符),可以使用 <![CDATA[]]> 把字面值包裹起来

1
2
3
4
5
6
7
8
9
10
11
12
13
14
public class UserDaoImpl implements UserDao {
private String company;
private int age;
public void setCompany(String company) {
this.company = company;
}
public void setAge(int age) {
this.age = age;
}
public void save() {
System.out.println(company+"==="+age);
System.out.println("UserDao save method running....");
}
}
1
2
3
4
<bean id="userDao" class="cn.jyw.dao.impl.UserDaoImpl">
<property name="company" value="哈哈哈哈"></property>
<property name="age" value="15"></property>
</bean>

配置 java.util.List 类型的属性, 需要指定 <list> 标签, 在标签里包含一些元素.

这些标签可以通过 <value> 指定简单的常量值, 通过 <ref> 指定对其他 Bean 的引用

通过<bean> 指定内置 Bean 定义. 通过 <null/> 指定空元素. 甚至可以内嵌其他集合。

数组的定义和 List 一样, 都使用 <list>

配置 java.util.Set 需要使用 <set> 标签, 定义元素的方法与 List 一样

1
2
3
4
5
6
7
8
9
10
public class UserDaoImpl implements UserDao {
private List<String> strList;
public void setStrList(List<String> strList) {
this.strList = strList;
}
public void save() {
System.out.println(strList);
System.out.println("UserDao save method running....");
}
}
1
2
3
4
5
6
7
8
9
<bean id="userDao" class="cn.jyw.dao.impl.UserDaoImpl">
<property name="strList">
<list>
<value>aaa</value>
<value>bbb</value>
<value>ccc</value>
</list>
</property>
</bean>

集合数据类型对象(List)的注入

1
2
3
4
5
6
7
8
9
10
public class UserDaoImpl implements UserDao {
private List<User> userList;
public void setUserList(List<User> userList) {
this.userList = userList;
}
public void save() {
System.out.println(userList);
System.out.println("UserDao save method running....");
}
}
1
2
3
4
5
6
7
8
9
10
<bean id="u1" class="cn.jyw.pojo.User"/>
<bean id="u2" class="cn.jyw.pojo.User"/>
<bean id="userDao" class="cn.jyw.dao.impl.UserDaoImpl">
<property name="userList">
<list>
<ref bean="u1"/>
<ref bean="u2"/>
</list>
</property>
</bean>

Java.util.Map 通过 <map> 标签定义, <map> 标签里可以使用多个 <entry> 作为子标签. 每个条目包含一个键和一个值.。

必须在 <key> 标签里定义键。

因为键和值的类型没有限制, 所以可以自由地为它们指定 <value>, <ref>, <bean><null> 元素.。

可以将 Map 的键和值作为 <entry> 的属性定义: 简单常量使用 key 和 value 来定义; Bean 引用通过 key-ref 和 value-ref 属性定义。

使用 <props> 定义 java.util.Properties, 该标签使用多个 作为子标签。 每个 标签必须定义 key 属性

1
2
3
4
5
6
7
8
9
10
11
public class UserDaoImpl implements UserDao {
private Map<String,User> userMap;
public void setUserMap(Map<String, User> userMap) {
this.userMap = userMap;
}
public void save() {
System.out.println(userMap);
System.out.println("UserDao save method running....");
}
}

1
2
3
4
5
6
7
8
9
10
<bean id="u1" class="cn.jyw.pojo.User"/>
<bean id="u2" class="cn.jyw.pojo.User"/>
<bean id="userDao" class="cn.jyw.dao.impl.UserDaoImpl">
<property name="userMap">
<map>
<entry key="user1" value-ref="u1"/>
<entry key="user2" value-ref="u2"/>
</map>
</property>
</bean>
1
2
3
4
5
6
7
8
9
10
public class UserDaoImpl implements UserDao {
private Properties properties;
public void setProperties(Properties properties) {
this.properties = properties;
}
public void save() {
System.out.println(properties);
System.out.println("UserDao save method running....");
}
}
1
2
3
4
5
6
7
8
9
<bean id="userDao" class="cn.jyw.dao.impl.UserDaoImpl">
<property name="properties">
<props>
<prop key="p1">aaa</prop>
<prop key="p2">bbb</prop>
<prop key="p3">ccc</prop>
</props>
</property>
</bean>

3.5 Bean的自动装配

Spring IOC 容器可以自动装配 Bean。需要做的仅仅是在 <bean>的 autowire 属性里指定自动装配的模式

  1. byType(根据类型自动装配): 若 IOC 容器中有多个与目标 Bean 类型一致的 Bean. 在这种情况下, Spring 将无法判定哪个 Bean 最合适该属性, 所以不能执行自动装配
  2. byName(根据名称自动装配): 必须将目标 Bean 的名称和属性名设置的完全相同。
  3. constructor(通过构造器自动装配): 当 Bean 中存在多个构造器时, 此种自动装配方式将会很复杂. 不推荐使用
  4. 在 Bean 配置文件里设置 autowire 属性进行自动装配将会装配 Bean 的所有属性. 然而, 若只希望装配个别属性时, autowire 属性就不够灵活了
  5. autowire 属性要么根据类型自动装配, 要么根据名称自动装配, 不能两者兼而有之
  6. 一般情况下,在实际的项目中很少使用自动装配功能,因为和自动装配功能所带来的好处比起来,明确清晰的配置文档更有说服力一些

3.6 Bean 的作用域

  1. 在 Spring 中, 可以在 <bean> 元素的 scope 属性里设置 Bean 的作用域。

  2. 默认情况下, Spring 只为每个在 IOC 容器里声明的 Bean 创建唯一一个实例

    整个 IOC 容器范围内都能共享该实例,所有后续的 getBean() 调用和 Bean 引用都将返回这个唯一的 Bean 实例

    该作用域被称为 singleton, 它是所有 Bean 的默认作用域。

3.7 使用外部属性文件

  1. 在配置文件里配置 Bean 时, 有时需要在 Bean 的配置里混入系统部署的细节信息(例如: 文件路径, 数据源配置信息等)。 而这些部署细节实际上需要和 Bean 配置相分离。

  2. 可通过 < context:property-placeholder> 元素来加载属性文件(.properties)中的属性值

  3. Spring 允许在value属性中使用 ${propName},以实现属性之间的相互引用

​ <context:property-placeholder location=”classpath:jdbc.properties”/>

1
2
3
4
5
6
7
8
9
10
11
<!--导入属性文件-->
<context : property-placeholder location= "classpath:db.properties"/>

<bean id= "dataSource" class= "com.mchange.v2.c3p0.ComboPooledDataSource">
<!--使用外部化属性文件的属性-->
<property name= "user" value="${user}"/>
<property name= "password" value="${password}"/>
<property name= "driverClass" value="${driverclass}"/>
<property name= "jdbcUrl" value="${jdbcurl}"/>
</bean>

3.8 引入其他配置文件(分模块开发)

实际开发中,Spring的配置内容非常多,这就导致Spring配置很繁杂且体积很大,所以,可以将部分配置拆解到其他配置文件中,而在Spring主配置文件通过import标签进行加载

1
<import resource="applicationContext-xxx.xml"/>

3.9 Bean 的生命周期

Spring IOC容器可以管理Bean的生命周期, Spring 允许在 Bean 生命周期的特定点执行定制的任务。

Spring IOC 容器对 Bean 的生命周期进行管理的过程:

  1. 通过构造器或工厂方法创建 Bean 实例
  2. 为 Bean 的属性设置值和对其他 Bean 的引用
  3. 调用 Bean 的初始化方法
  4. Bean 可以使用了
  5. 当容器关闭时, 调用 Bean 的销毁方法
  6. 在Bean的声明里设置 init-method 和 destroy-method 属性, 为 Bean 指定初始化和销毁方法

Bean的完整生命周期经历了各种方法调用,这些方法可以划分为以下几类:

  1. Bean自身的方法:
    这个包括了Bean本身调用的方法和通过配置文件中<bean>的init-method和destroy-method指定的方法。
  2. Bean级生命周期接口方法:  
    这个包括了BeanNameAwareBeanFactoryAwareInitializingBeanDiposableBean这些接口的方法。
  3. 容器级生命周期接口方法:
    这个包括了InstantiationAwareBeanPostProcessorBeanPostProcessor 这两个接口实现,一般称它们的实现类为“后处理器”。
  4. 工厂后处理器接口方法:
    这个包括了AspectJWeavingEnabler, ConfigurationClassPostProcessor, CustomAutowireConfigurer等等非常有用的工厂后处理器接口的方法。工厂后处理器也是容器级的。在应用上下文装配配置文件之后立即调用

创建 Bean 后置处理器

  1. Bean 后置处理器允许在调用初始化方法前后对 Bean 进行额外的处理

  2. Bean 后置处理器对 IOC 容器里的所有 Bean 实例逐一处理, 而非单一实例.

    其典型应用是: 检查 Bean 属性的正确性或根据特定的标准更改 Bean 的属性。

  3. 对Bean 后置处理器而言, 需要实现接口(BeanPostProcessor)

    在初始化方法被调用前后, Spring 将把每个 Bean 实例分别传递给上述接口的以下两个方法:

    Object postProcessAfterInitialization (Object bean,String bname)

    Object postProcessBeforeInitialization (Object bean,String bname)

添加 Bean 后置处理器后 Bean 的生命周期
Spring IOC 容器对 Bean 的生命周期进行管理的过程:

  1. 通过构造器或工厂方法创建 Bean 实例;

  2. 为 Bean 的属性设置值和对其他 Bean 的引用;

  3. 将Bean例传递给Bean后置处理器的 postProcessBeforeInitialization 方法;

  4. 调用 Bean 的初始化方法;

  5. 将Bean实例传递给Bean后置处理器的postProcessAfterInitialization方法;

  6. Bean 可以使用了

  7. 当容器关闭时, 调用 Bean 的销毁方法

后置处理器主要应用在spring容器的底层,spring容器的原生AOP要使用它实现对Bean的切面编程

Spring的重点配置

1
2
3
4
5
6
7
8
9
10
11
12
13
<bean>标签
id属性:在容器中Bean实例的唯一标识,不允许重复
class属性:要实例化的Bean的全限定名
scope属性:Bean的作用范围,常用是Singleton(默认)和prototype
<property>标签:属性注入
name属性:属性名称
value属性:注入的普通属性值
ref属性:注入的对象引用值
<list>标签
<map>标签
<properties>标签
<constructor-arg>标签
<import>标签:导入其他的Spring的分文件

4. Spring相关API

4.1 ApplicationContext的继承体系

ClassPathXmlApplicationContext

Spring 提供了两种类型的 IOC 容器实现:

  • BeanFactory: IOC 容器的基本实现,是 Spring 框架的基础设施,面向 Spring 本身

  • ApplicationContext:供了更多的高级特性. 是 BeanFactory 的子接口,面向使用 Spring 框架的开发者,几乎所有的应用场合都直接使用 ApplicationContext 而非底层的 BeanFactory,无论使用何种方式, 配置文件是相同的

4.2 ApplicationContext的实现类

applicationContext:接口类型,代表应用上下文,可以通过其实例获得 Spring 容器中的 Bean 对象

  1. ClassPathXmlApplicationContext

    它是从类的根路径下加载配置文件 推荐使用这种

  2. FileSystemXmlApplicationContext

    它是从磁盘路径上加载配置文件,配置文件可以在磁盘的任意位置。

  3. AnnotationConfigApplicationContext

    当使用注解配置容器对象时,需要使用此类来创建 spring 容器。它用来读取注解。

  4. ConfigurableApplicationContext

    扩展于 ApplicationContext,新增加两个主要方法:refresh() 和 close(), 让 ApplicationContext 具有启动、刷新和关闭上下文的能力

  5. WebApplicationContext

    是专门为**WEB**应用而准备的,它允许从相对于 WEB 根目录的路径中完成初始化工作

4.3 getBean()方法使用

getBean()的源码

1
2
3
4
5
6
7
8
public Object getBean(String name) throws BeansException {
assertBeanFactoryActive();
return getBeanFactory().getBean(name);
}
public <T> T getBean(Class<T> requiredType) throws BeansException {
assertBeanFactoryActive();
return getBeanFactory().getBean(requiredType);
}

其中,当参数的数据类型是字符串时,表示根据Bean的id从容器中获得Bean实例,返回是Object,需要强转。 当参数的数据类型是Class类型时,表示根据类型从容器中匹配Bean实例,当容器中相同类型的Bean有多个时, 则此方法会报错

1
2
3
4
5
ApplicationContext applicationContext = new 
ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService1 = (UserService)
applicationContext.getBean("userService");
UserService userService2 = applicationContext.getBean(UserService.class);

4.4 Spring的重点API

1
2
3
ApplicationContext app = new ClasspathXmlApplicationContext("xml文件")
app.getBean("id")
app.getBean(Class)

当容器中某一类型的对象有多个 用不同的id获取

有一个可以用.class获取