前言:一直想写一个关于Spring的系列文章,但是构思许久却不知道以什么方式阐述,毕竟要把一个复杂框架说清楚并不是那么容易的,我也只能尽力而为了。Spring系列文章打算以这样的顺序展开:以思维导图的方式统筹介绍Spring -> 介绍Spring基本用法 -> 深入使用Spring -> 窥探Spring源码 -> 如有必要以思维导图总结概述Spring。本文介绍Spring基本用法,目的更多的是入门Spring的使用方法,demo、附件项目更是快速查询Spring基本用法的工具。
本篇文章重点关注以下问题:
- 概述Spring核心功能
- IOC、DI基本概念
- 两种依赖注入的方式
1. Spring核心功能
Spring是一个非常活跃的开源框架,它以一种非侵入式的方式来管理你的代码,Spring提倡”最少侵入”,甚至在项目开发过程中,你完全没感到Spring在发挥作用。
Spring框架的核心功能有两个:
- 一是Spring容器作为超级工厂,负责创建、管理所有的java对象,这些java对象称为Bean。(IOC实现)
- 二是Spring容器管理Bean之间的依赖关系。(DI实现)
2. IOC、DI基本概念
IOC:即控制反转,把对象的创建、初始化、销毁等工作交给spring容器来做,由spring容器控制对象的生命周期。
* 控制:就是指对对象的创建、维护、销毁等生命周期的控制,这个过程一般是由我
们的程序去主动控制的,如使用new关键字去创建一个对象(创建),在使
用过程中保持引用(维护),在失去全部引用后由GC去回收对象(销
毁)。
* 反转:就是指对对象的创建、维护、销毁等生命周期的控制由程序控制改为由IOC
容器控制,需要某个对象时就直接通过名字去IOC容器中获取。
DI:即依赖注入,是IOC的一种重要实现,与IOC描述的是同一个概念。
一个对象的创建往往会涉及到其他对象的创建,比如一个对象A的成员变量持有着另一个对象B的引用,这就是依赖,A依赖于B。IOC机制既然负责了对象的创建,那么这个依赖关系也就必须由IOC容器负责起来。负责的方式就是DI——依赖注入,通过将依赖关系写入配置文件,然后在创建有依赖关系的对象时,由IOC容器注入依赖的对象,如在创建A时,检查到有依赖关系,IOC容器就把A依赖的对象B创建后注入到A中(组装,通过反射机制实现),然后把A返回给对象请求者,完成工作。
其工作过程如下图所示:
对应于Spring的两个核心功能,程序员所要做的工作就两点:
- 一是开发Bean:Spring建议面向接口编程,java对象间的耦合仅在接口层面,无需关心服务的具体实现;
- 二是配置Bean:通过xml配置文件或注解的方式告知Spring,需要Spring管理哪些Bean,以及Bean之间的依赖关系;
在需要使用Bean时,直接从容器中去拿即可。
3. 依赖注入的方式
依赖注入(DI)通常有如下两种方式:
- 设值注入:IOC容器使用成员变量的setter方法来注入被依赖对象;
- 构造注入:IOC容器使用构造器来注入被依赖对象。
3.1 设值注入
设值注入:IOC容器使用成员变量的setter方法来注入被依赖对象。
开发过程如下:(此处以xml配置文件的形式说明)
* Pen.java:用于实现签名功能;
* Person.java:需要使用Pen来签名,即Person.java依赖于Pen.java
实现服务类功能:
* BallpointPen.java:提供用圆珠笔实现签名的功能;
* Pencil.java:提供用铅笔实现签名的功能;
* Chinese.java:Person的子类,表示中国人来签名;
Chinese.java代码如下,类中需要定义Pen接口对象的成员变量pen,并提供其setter方法,用于设值注入:
package com.wj.chapter1.ioc.setter.xml.Service;
import com.wj.chapter1.ioc.setter.xml.IService.Pen;
import com.wj.chapter1.ioc.setter.xml.IService.Person;
public class Chinese implements Person {
private Pen pen; /** 代码耦合的层次仅限于接口层次. */
// 设值需要的setter方法
public void setPen(Pen pen) {
this.pen = pen;
}
@Override
public void signOwnName() {
pen.signName("中国人");
}
}
步骤二:配置Bean
通过xml配置文件的方式告知Spring,需要Spring管理哪些Bean,以及Bean之间的依赖关系。
xml配置文件中,每个Bean对应Spring容器里的一个java实例,property节点元素指明Bean之间的依赖关系。
步骤三:调用Bean
使用前面定义的服务,实现中国人用铅笔签名的功能:
package com.wj.chapter1.ioc.setter.xml.main;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import com.wj.chapter1.ioc.setter.xml.IService.Person;
public class Main {
// 1.指明xml配置文件位置,便于Spring读取,从而知道Bean的相关信息
private static final String PATH_XML = "com/wj/chapter1/ioc/setter/xml/applicationContext-setter.xml";
@SuppressWarnings("resource")
public static void main(String[] args) {
// 2.根据xml配置文件,创建Spring IOC容器的上下文
ApplicationContext cxt = new ClassPathXmlApplicationContext(PATH_XML);
// 3.在需要使用服务的时候,直接从Spring容器中获取
Person chinese = cxt.getBean("chinese", Person.class);
// 4.获取提供服务的对象后,调用Bean的服务
chinese.signOwnName();
}
}
步骤四:查看调用结果
3.2 构造注入
构造注入:IOC容器使用构造器来注入被依赖对象。
开发过程如下:(此处以xml配置文件的形式说明)
步骤一:开发Bean
定义服务类接口:
* Pen.java:用于实现签名功能;
* Person.java:需要使用Pen来签名,即Person.java依赖于Pen.java
实现服务类功能:
* BallpointPen.java:提供用圆珠笔实现签名的功能;
* Pencil.java:提供用铅笔实现签名的功能;
* Chinese.java:Person的子类,表示中国人来签名;
Chinese.java代码如下,类中需要定义Pen接口对象的成员变量pen,并提供带参数的构造函数,用于构造注入:
package com.wj.chapter1.ioc.constructor.xml.Service;
import com.wj.chapter1.ioc.constructor.xml.Iservice.Pen;
import com.wj.chapter1.ioc.constructor.xml.Iservice.Person;
public class Chinese implements Person {
private Pen pen; /** 代码耦合的层次仅限于接口层次. */
// 构造注入所需的带参数的构造器
public Chinese(Pen pen) {
this.pen = pen;
}
@Override
public void signOwnName() {
pen.signName("中国人");
}
}
步骤二:配置Bean
通过xml配置文件的方式告知Spring,需要Spring管理哪些Bean,以及Bean之间的依赖关系。
步骤三:调用Bean
使用前面定义的服务,实现中国人用圆珠笔签名的功能:
package com.wj.chapter1.ioc.constructor.xml.main;
import org.springframework.context.ApplicationContext;
import org.springframework.context.support.ClassPathXmlApplicationContext;
import com.wj.chapter1.ioc.constructor.xml.Iservice.Person;
public class Main {
// 1.指明xml配置文件位置,便于Spring读取,从而知道Bean的相关信息
private static final String PATH_XML = "com/wj/chapter1/ioc/constructor/xml/applicationContext-constructor.xml";
@SuppressWarnings("resource")
public static void main(String[] args) {
// 2.根据xml配置文件,创建Spring IOC容器的上下文
ApplicationContext cxt = new ClassPathXmlApplicationContext(PATH_XML);
// 3.在需要使用服务的时候,直接从Spring容器中获取
Person chinese = cxt.getBean("chinese", Person.class);
// 4.获取提供服务的对象后,调用Bean的服务
chinese.signOwnName();
}
}
步骤四:查看调用结果
3.3 两种注入方式的对比
建议采用以设值注入为主,构造注入为辅的注入策略。对于依赖关系无需变化的注入,可以考虑采用构造注入,其他依赖关系的注入,则尽量用设值注入。