Spring简介
学习目标
- Spring框架设计思想
- 基础操作,思考操作与思想间的联系
- 熟悉应用操作的同时,体会思想
Spring家族
Spring体系结构
Spring核心概念
当前代码编写存在的问题
- 主要痛点:耦合度偏高
- 解决方案:使用对象时,在程序中不要主动使用new产生对象,转换为由外部提供对象
程序的耦合
- 耦合:程序间的依赖关系
- 类之间的依赖
- 方法间的依赖
- 解耦:降低程序间的依赖关系
注意:编译期不依赖,运行期依赖。
- 解耦思路:
step1. 使用反射来创建对象,而避免使用new关键字
step2. 通过读取配置文件来获取要创建的对象全限定类名
工厂模式解耦
创建Bean对象的工厂
- Bean:可重用组件
- JavaBean用Java语言编写的可重用组件
JavaBean>实体类
- 作用:创建service和dao对象
- 需要一个配置文件来配置我们的service和dao,配置内容:唯一标识=全限定类名(key=value)
- 通过读取配置文件(.xml或.properties)中的配置内容,反射创建对象
import java.io.InputStream;
import java.util.Enumeration;
import java.util.HashMap;
import java.util.Map;
import java.util.Properties;
public class BeanFactory {
//定义一个Properties对象
private static Properties props;
//定义一个Map,用于存放我们要创建的对象。我们把它称之为容器
private static Map<String,Object> beans;
//使用静态代码块为Properties对象赋值
static {
try {
//实例化对象
props = new Properties();
//获取properties文件的流对象
InputStream in = BeanFactory.class.getClassLoader().getResourceAsStream("bean.properties");
props.load(in);
//实例化容器
beans = new HashMap<String,Object>();
//取出配置文件中所有的Key
Enumeration keys = props.keys();
//遍历枚举
while (keys.hasMoreElements()){
//取出每个Key
String key = keys.nextElement().toString();
//根据key获取value
String beanPath = props.getProperty(key);
//反射创建对象
Object value = Class.forName(beanPath).newInstance();
//把key和value存入容器中
beans.put(key,value);
}
}catch(Exception e){
throw new ExceptionInInitializerError("初始化properties失败!");
}
}
/**
* 根据bean的名称获取对象
* @param beanName
* @return
*/
public static Object getBean(String beanName){
return beans.get(beanName);
}
/**
* 根据Bean的名称获取bean对象
* @param beanName
* @return
public static Object getBean(String beanName){
Object bean = null;
try {
String beanPath = props.getProperty(beanName);
// System.out.println(beanPath);
bean = Class.forName(beanPath).newInstance();//每次都会调用默认构造函数创建对象
}catch (Exception e){
e.printStackTrace();
}
return bean;
}*/
}
核心概念
IOC
降低程序间的耦合(依赖关系)
- IOC(Inversion of Control)控制反转:
使用对象时,由主动new产生对象转换为由外部提供对象,此过程中对象创建控制权由程序转移到外部(将new对象的权力交给Spring,直接从Spring中获取对象使用)。 - Spring技术对IOC思想进行了实现
- Spring提供了一个容器,称为IOC容器,用来充当IOC思想中的“外部”。
- IOC容器负责对象的创建,初始化等一系列工作,被创建或被管理的对象在IOC容器中统称为Bean。
作用:获取spring容器的Ioc核心容器,并根据id获取对象
ApplicationContext的三个常用实现类
- ClassPathXmlApplicationContext:可以加载类路径下的配置文件,要求配置文件必须在类路径下。(常用)
- FileSystemXmlApplicationContext:可以加载磁盘任意路径下的配置文件(必须拥有放文权限)
- AnnotationConfigApplicationContext:当我们使用注解配置容器对象时,需要使用此类来创建 spring 容器。它用来读取注解。
核心容器的两个接口引发出的问题
- ApplicationContext:单例对象适用
在构建核心容器时,创建对象采用的策略是采用立即加载的方式,即为只要一读取完配置文件马上就创建配置文件中配置的对象
- BeanFactory:多例对象适用
在构建核心容器时,创建对象采用的策略时采用延迟加载的方式,即为什么时候根据id获取对象了,什么时候才真正的创建对象
DI
- DI(Dependency Injection)依赖注入:
在容器中建立bean与bean之间的依赖关系的整个过程。 - 目标:充分解耦
- 使用IOC容器管理bean(IOC)
- 在IOC容器内将有依赖关系的bean进行关系绑定(DI)
- 最终效果:
使用对象时不仅可以直接从IOC容器中获取,并且获取到bean已经绑定了所有的依赖关系
IOC和DI入门案例
IOC入门案例
> <bean>
标签中id属性和class属性的作用
思路分析
- 管理什么(Service和Dao)
- 如何将被管理的对象告知IOC容器(配置文件)
- 被管理的对象交给IOC容器,如何获取到IOC容器(接口)
- IOC容器得到后,如何从容器中获取bean(接口方法)
- 使用Spring导入哪些坐标(pom.xml)
实现步骤
step1. 导入Spring坐标
step2. 定义Spring管理的类(接口)
step3. 创建Spring配置文件,配置对应类作为Spring管理的bean对象
step4. 初始化IOC容器(Spring核心容器/Spring容器),通过容器获取bean对象
代码实现
step1.
<dependencies>
<!--导入spring的坐标spring-context,对应版本是5.2.10.RELEASE-->
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.2.10.RELEASE</version>
</dependency>
</dependencies>
step2.
- BookDao接口和BookDaoImpl实现类
public interface BookDao {
public void save();
}
public class BookDaoImpl implements BookDao {
public void save() {
System.out.println("book dao save ...");
}
}
- BookService接口和BookServiceImpl实现类
public interface BookService {
public void save();
}
public class BookServiceImpl implements BookService {
private BookDao bookDao = new BookDaoImpl();
public void save() {
System.out.println("book service save ...");
bookDao.save();
}
}
- step3.
- 定义applicationContext.xml配置文件并配置BookServiceImpl
<?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 http://www.springframework.org/schema/beans/spring-beans.xsd">
<!--
bean标签:表示配置bean
id属性:表示给bean起名字
class属性:表示给bean定义类型
-->
<bean id="bookService" class="com.itheima.service.impl.BookServiceImpl"></bean>
</beans>
注意事项:bean定义时id属性在同一个上下文中(IOC容器中)不能重复
- step4.
public class App {
public static void main(String[] args) {
//1.创建IoC容器对象,加载spring核心配置文件
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
//2 从IOC容器中获取Bean对象(BookService对象)
BookService bookService= (BookService)ctx.getBean("bookService");
//3 调用Bean对象(BookService对象)的方法
bookService.save();
}
}
DI入门案例
<property>
标签中name属性和ref属性的作用
DI思路分析
- 基于IOC管理bean
- Service中使用new形式创建的Dao对象是否保留(否)
- Service中需要的Dao对象如何进行到Service中(提供方法)
- Service与Dao间的关系如何描述(配置)
实现步骤
step1. 删除使用new的形式创建对象的代码
step2.提供依赖对象对应的setter方法
step3.配置Service与Dao之间的关系
代码实现
- step1.
public class BookServiceImpl implements BookService {
private BookDao bookDao; //【第一步】删除使用new的形式创建对象的代码
public void save() {
System.out.println("book service save ...");
bookDao.save();
}
}
- step2.
public class BookServiceImpl implements BookService {
private BookDao bookDao;
public void save() {
System.out.println("book service save ...");
bookDao.save();
}
//【第二步】提供依赖对象对应的setter方法
public void setBookDao(BookDao bookDao) {
this.bookDao = bookDao;
}
}
- step3.
在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 http://www.springframework.org/schema/beans/spring-beans.xsd">
<!--
bean标签:表示配置bean
id属性:表示给bean起名字
class属性:表示给bean定义类型
-->
<bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl"/>
<bean id="bookService" class="com.itheima.service.impl.BookServiceImpl">
<!--配置server与dao的关系
property标签:表示配置当前bean的属性
name属性:表示配置哪一个具体的属性
ref属性:表示参照哪一个bean
-->
<property name="bookDao" ref="bookDao"/>
</bean>
</beans>
Bean的基础配置
- 在
<bean>
标签上如何配置别名 - Bean的默认作用范围,修改方法
Bean基础配置
Bean别名配置
* 注意事项获取bean无论是通过id还是bean获取,如果无法获取到,将抛出异常NoSuchBeanDefinitionException
NoSuchBeanDefinitionException:No bean named ‘bookeServiceImpl’ available
Bean作用范围配置
提示:scope的取值不仅仅只有singleton和prototype,还有request、session、application、websocket,表示创建出的对象放置在web容器(tomcat)对应位置
Bean的实例化
spring对bean的管理细节
- 创建bean的三种方式
- bean对象的作用范围
- bean对象的生命周期
Bean创建方法
Bean本质为对象,创建Bean使用构造方法。
实例化Bean的三种方式
构造方法方式
使用默认构造函数创建
- 在spring的配置文件中使用bean标签,配以id和class属性之后,且没有其他属性和标签时。
- 采用的就是默认构造函数创建bean对象,此时如果类中没有默认构造函数,则对象无法创建。
- BookDaoImpl实现类
public class BookDaoImpl implements BookDao {
public BookDaoImpl() {
System.out.println("book dao constructor is running ....");
}
public void save() {
System.out.println("book dao save ...");
}
}
- applicationContext.xml配置
<!--把对象的创建交给spring来管理-->
<!--方式一:构造方法实例化bean-->
<bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl"/>
- AppForInstanceBook测试类
public class AppForInstanceBook {
public static void main(String[] args) {
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
BookDao bookDao = (BookDao) ctx.getBean("bookDao");
bookDao.save();
}
}
静态工厂方式
- 使用工厂中的静态方法创建对象(使用某个类中的静态方法创建对象,并存入spring容器)
- OrderDao接口和OrderDaoImpl实现类
public interface OrderDao {
public void save();
}
public class OrderDaoImpl implements OrderDao {
public void save() {
System.out.println("order dao save ...");
}
}
- OrderDaoFatory工厂类
//静态工厂创建对象
public class OrderDaoFactory {
public static OrderDao getOrderDao(){
System.out.println("factory setup....");
return new OrderDaoImpl();
}
}
- applicationContext.xml配置
<!--方式二:使用静态工厂实例化bean-->
<bean id="orderDao" class="com.itheima.factory.OrderDaoFactory" factory-method="getOrderDao"/>
- AppForInstanceOrder测试类
public class AppForInstanceOrder {
public static void main(String[] args) {
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
OrderDao orderDao = (OrderDao) ctx.getBean("orderDao");
orderDao.save();
}
}
实例工厂方式
- 使用普通工厂中的方法创建对象(使用某个类中的方法创建对象,并存入spring容器)
-
- UserDao接口和UserDaoImpl实现类
public interface UserDao {
public void save();
}
public class UserDaoImpl implements UserDao {
public void save() {
System.out.println("user dao save ...");
}
}
-
- UserDaoFactory工厂类
//实例工厂创建对象
public class UserDaoFactory {
public UserDao getUserDao(){
return new UserDaoImpl();
}
}
-
- applicationContext.xml配置
<!--方式三:使用实例工厂实例化bean-->
<bean id="userFactory" class="com.itheima.factory.UserDaoFactory"/>
<bean id="userDao" factory-method="getUserDao" factory-bean="userFactory"/>
-
- AppForInstanceUser测试类
public class AppForInstanceUser {
public static void main(String[] args) {
// //创建实例工厂对象
// UserDaoFactory userDaoFactory = new UserDaoFactory();
// //通过实例工厂对象创建对象
// UserDao userDao = userDaoFactory.getUserDao();
// userDao.save();
ApplicationContext ctx = new ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userDao = (UserDao) ctx.getBean("userDao");
userDao.save();
}
}
bean的作用范围调整
bean标签的scope属性:
- 作用:用于指定bean的作用范围
- 取值: 常用的就是单例的和多例的
singleton:单例的(默认值)
prototype:多例的
request:作用于web应用的请求范围
session:作用于web应用的会话范围
global-session:作用于集群环境的会话范围(全局会话范围),当不是集群环境时,它就是session
Bean的生命周期
- 多例的Bean能够配置并执行销毁的方法
- 如何执行Bean销毁的方法
生命周期相关概念介绍
- 生命周期:从创建到消亡的完整过程
- Bean生命周期:Bean从创建到销毁的完整过程
- Bean生命周期控制:在Bean创建后到销毁前做一些事情
Bean销毁时机
- 容器关闭前触发Bean的销毁
- 关闭容器方式:
- 手工关闭容器:
ConfigurableApplicationContext
接口close()
操作
- 注册关闭钩子,在虚拟机退出前先关闭容器再退出虚拟机:
ConfigurableApplicationContext
接口registerShutdownHook()
操作
依赖注入(DI配置) Dependency Injection
- 概念:对程序间依赖关系的维护
依赖注入方式
依赖注入的两种方式
- setter注入:简单类型、引用类型
- 构造器注入:简单类型、引入类型
setter方式注入
- setter方式注入使用的子标签:
- 配置中使用
property
标签ref属性注入引用类型对象- 在Bean中定义引用类型属性并提供可访问的set方法
- 配置中使用
property
标签value属性注入简单类型数据
- 标签:property
- 位置:bean标签的内部
- 标签的属性
name
:用于指定注入时所调用的set方法名称value
:用于提供基本类型和String类型的数据ref
:用于指定其他的bean类型数据。它指的就是在spring的Ioc核心容器中出现过的bean对象
- 优势:
创建对象时没有明确的限制,可以直接使用默认构造函数 - 劣势:
如果有某个成员必须有值,则获取对象是有可能setter方法没有执行。
引用类型
简单类型
构造方式注入
- 构造方式注入使用的子标签:
- 在Bean中定义引用类型属性并提供可访问的构造方法
- 配置中使用
constructor-arg
标签ref属性注入引用类型对象- 在Bean中定义引用类型属性并提供可访问的set方法
- 配置中使用
constructor-arg
标签value属性注入简单类型数据
-
标签:constructor-arg
-
位置:bean标签的内部
-
标签中的属性
type
:用于指定要注入的数据的数据类型,该数据类型也是构造函数中某个或某些参数的类型index
:用于指定要注入的数据给构造函数中指定索引位置的参数赋值。索引的位置是从0开始name
:用于指定给构造函数中指定名称的参数赋值 常用的
以上三个用于指定给构造函数中哪个参数赋值value
:用于提供基本类型和String类型的数据ref
:用于指定其他的bean类型数据。它指的就是在spring的Ioc核心容器中出现过的bean对象
-
优势:
在获取bean对象时,注入数据是必须的操作,否则对象无法创建成功。 -
劣势:
改变了bean对象的实例化方式,使我们在创建对象时,即使用不到这些数据,也必须提供。
引用类型
简单类型
复杂类型/集合类型的注入
- 用于给List结构集合注入的标签:
list array set
- 用于个Map结构集合注入的标签:
map props
结构相同,标签可以互换
依赖注入方式选择
- 强制依赖使用构造器进行,使用setter注入有概率不进行注入导致null对象出现
- 可选依赖使用setter注入进行,灵活性强
- Spring框架倡导使用构造器,第三方框架内部大多数采用构造器注入的形式进行数据初始化,相对严谨
- 如果有必要可以两者同时使用,使用构造器注入完成强制依赖的注入,使用setter注入完成可选依赖的注入
- 实际开发过程中还要根据实际情况分析,如果受控对象没有提供setter方法就必须使用构造器注入
- 自己开发的模块主要使用setter注入
依赖自动装配
自动装配
- 概念:IOC容器根据Bean所依赖的资源在容器中自动寻找并注入到Bean中的过程
- 自动装配方式:
.按类型
按名称
按构造方法
不启用自动装配
自动装配类型
依赖自动装配
配置中使用bean标签autowire属性设置自动装配的类型
<bean id="bookDao" class="com.itheima.dao.impl.BookDaoImpl"/>
<bean id="bookService" class="com.itheima.service.impl.BookServiceImpl" autowire="byType"/>
依赖自动装配特征
- 自动装配用于引用类型依赖注入,不能对简单类型进行操作
- 使用按类型装配时(byType)必须保障容器中相同类型的bean唯一,推荐使用
- 使用按名称装配时(byName)必须保障容器中具有指定名称的bean,因变量名与配置耦合,不推荐使用
- 自动装配优先级低于setter注入与构造器注入,同时出现时自动装配配置失效
集合注入
注入数组类型数据
<property name="array">
<array>
<value>100</value>
<value>200</value>
<value>300</value>
</array>
</property>
注入List类型数据
<property name="list">
<list>
<value>itcast</value>
<value>itheima</value>
<value>boxuegu</value>
<value>chuanzhihui</value>
</list>
</property>
注入Set类型数据
<property name="set">
<set>
<value>itcast</value>
<value>itheima</value>
<value>boxuegu</value>
<value>boxuegu</value>
</set>
</property>
注入Map类型数据
<property name="map">
<map>
<entry key="country" value="china"/>
<entry key="province" value="henan"/>
<entry key="city" value="kaifeng"/>
</map>
</property>
注入Properties类型数据
<property name="properties">
<props>
<prop key="country">china</prop>
<prop key="province">henan</prop>
<prop key="city">kaifeng</prop>
</props>
</property>
说明:property标签表示setter方式注入,构造方式注入constructor-arg标签内部也可以写<array>、<list>、<set>、<map>、<props>标签
单例与多例(补充)
单例模式
- 概念:一个类有且仅有一个实例,并且自行实例化向整个系统提供。
多例模式(反面模式)
- 概念:指存在一个类有多个相同实例,而且该实例都是该类本身。
- 特点:
- 多例类可以有多个实例。
- 多例类必须自己创建、管理自己的实例,并向外界提供自己的实例。
单例模式与多例模式说明
- 单例模式和多例模式属于对象模式。
- 单例模式的对象在整个系统中只有一份,而多例模式可以有多个实例类。
- 它们都不对外提供构造方法,即构造方法都为私有。