SpringIoC&DI
1. Spring概述
1.1 什么是Spring(理解)
Spring是分层的JavaEE/SE应用全栈轻量级开源框架,以Ioc(Inverse Of Control:控制反转)和AOP(Aspect Oriented Programming:面向切面编程)为内核。
Spring提供了展示层Spring MVC 、持久层SpringJDBCTemplate以及业务层事务管理等众多企业级应用技术,还能整合开源世界众多第三方框架和类库,逐渐成为使用最多的JavaEE企业开应用开源框架。
1.2 Spring的发展历程(了解)
Rod Johnson(Spring之父)
2017年9月发布了Spring的最新版本Spring5.0通用版(GA)
1.3 Spring的优势(理解)
方便解耦,简化开发
AOP编程的支持
声明式事务的支持
方便程序测试
方便继承各种优秀的框架
降低JavaEE API的使用难度
Java源码是经典学习范例
1.4 Spring的体系结构(了解)
此处应该有图片
2. Spring快速入门
2.1 Spring开发程序的步骤
①导入spring开发的基本包坐标
②编写dao接口和实现类
③编写Spring核心配置文件(applicationContext.xml)
④在Spring核心配置文件中配置UserDaoImpl
⑤使用Spring的API获取Bean实例(UserDaoImpl的对象)
2.2 导入Spring开发的基本包坐标
<properties>
<spring.version>5.0.5.RELEASE</spring.version>
</properties>
<!--导入Spring-context包,context依赖于core、beans、expression-->
<dependencies>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>${spring.version}</version>
</dependency>
<!-- 需要单元测试,所以导入junit的坐标-->
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
</dependency>
</dependencies>
2.3 创建UserDao和UserDaoImpl类
public interface UserDao {
public void save();
}
public class UserDaoImpl implements UserDao {
@Override
public void save() {
System.out.println("UserDao save method running....");
}
}
2.4 创建核心配置文件并配置bean
在类路径(resources)下创建核心配置文件applicationContext.xml。
配置文件类型为spring config,名字任意,建议按照惯例写成applicationContext.xml。
在applicationContext.xml文件中配置Bean,关联UserDaoImpl类。
<?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 id="userDao" class="com.itheima.dao.impl.UserDaoImpl"></bean>
</beans>
2.5 *通过Spring的API获取bean实例并调用
*在test类里面编写Junit单元测试(开发环境使用业务层代码完成dao层代码调用)
@Test
public void userDaoTest() {
// 加载配置文件,创建Spring容器对象
ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userDao = (UserDao) ac.getBean("userDao");
userDao.saveUser(new User());
}
控制台输出如下结果:
UserDao save method running....
2.6 *总结对比(★理解★)
上面的内容就是通过Spring的Ioc思想,将UserDaoImpl的创建权由程序编写者交给了Spring容器,当然最终拿到对象并且使用对象的人肯定还是程序员,但是对象的创建者已经发生了改变。
学习到这里,有同学可能会有疑问,这样做原来很简单的一句代码:
UserDao userdao = new UserDaoImpl();
userDao.saveUser(new User());
竟然变成了复杂的代码+配置
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"></bean>
ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userDao = (UserDao) ac.getBean("userDao");
userDao.saveUser(new User());
这样出力不讨好为了什么?这里首先我们要明白,使用Spring框架的目的是什么?
1)使用Spring的IOC容器,将对象创建、对象之间的依赖关系管理(马上学)交给Spring,降低组件之间的耦合性,让我们更专注于应用逻辑。
2)还可以提供诸如展示层、持久层、事务管理和兼容诸多第三方开源框架等等优势。
所以,使用框架最终目的并不是为了让我们少写代码,是要简化Java开发,很直观的一点就是解耦合(大家先记住,之后会有体验),在另外一个层面更好的完成想要的功能。
当然,在某些时候,也会适当减少我们的代码量(在讲完DI之后会有体现)。
3. Spring配置文件的配置
3.1 *Bean标签的基本配置 - id&class
*Bean标签作用:明确 交给Spring创建的对象 所属的类
默认情况下,Spring容器通过反射调用上述类的无参构造创建该类对象,如果没有无参构造将创建不成功。
id:Bean所示类在Spring容器中的唯一标识
class:配置Bean实例所属类的全限定名称
3.2 Bean标签范围的配置 - scope
Bean标签有属性scope,用于表示bean所示对象的作用范围,取值如下
取值范围 | 说明 |
---|---|
singleton | 默认值,单例的。整个容器只有一个bean所示类的对象。 |
prototype | 多例的。整个Spring容器中可以有多个bean所示类的对象。 |
request | Web项目中,Spring创建一个Bean所示类的对象,存入request域中 |
session | Web项目中,Spring创建一个Bean所示类的对象,存入session域中 |
global session | Web项目中,在Portlet环境。如果没有Portlet环境,该配置等同于session |
- 当scope属性的取值为singleton时
Bean的实例化个数:1个
Bean的初始化时机:当Spring核心配置文件被加载时,实例化该Bean
Bean的生命周期:
对象的创建:当应用加载,容器创建时,对象就被创建了
对象的运行:只要容器在,对象就一直存在
对象的销毁:当应用卸载时,容器被销毁了,对象也一起被销毁
2)当scope属性的取值为prototype时
Bean的实例化个数:多个
Bean的初始化时机:当调用getBean()方法时,实例化Bean
Bean的生命周期:
对象的创建:当调用getBean()方法时,实例化Bean
对象的运行:只要对象一直在使用,对象就会一直存在
对象的销毁:当对象长时间不被使用时,被Java的垃圾回收器回收
3.3 Bean生命周期配置指定方法
init-method:指定Bean所示类初始化方法的名称
destroy-method:指定Bean所示类销毁方法的名称(需要在明确有容器销毁时才会被调用,具体容器实现类对象的close对象被调用时会调用该指定方法)
3.4 Bean实例化的三种方式
三种创建方式下,Spring提供的获取Bean对象的方法相同;仅仅是配置的方式以及配置方式背后的实现不同。获取bean的代码如下:
ApplicationContext ac = new ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userDao = (UserDao) ac.getBean("userDao");
userDao.saveUser(new User());
1)使用Bean所示类的无参构造实例化(重点)
Bean实例化最常用的方式,Spring核心配置文件配置如下:
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"></bean>
2)使用静态工厂方法实例化(应用场景:eg:connection的获取/解决系统遗留问题 )
Spring核心配置文件配置如下:
<bean id="userDao" class="com.itheima.factory.StaticFactory" factory-method="getUserDao"/>
需要编写StaticFactory工厂类,并提供static方法返回Bean所示类对象
package com.itheima.factory;
……
public class StaticFactory{
public satic UserDao getUserDao(){
return new UserDaoImpl();
}
}
3)使用实例工厂方法实例化(解决系统遗留问题 )
因为需要使用工厂实例(对象)创建Bean所示类的对象,所以需要为工厂创建一个Bean标签,Spring核心配置文件配置如下:
<bean id="dynamicFactory" class="com.itheima.factory.DynamicFactory"></bean>
<bean id="userDao" factory-bean="dynamicFactory" factory-method="getUserDao"></bean>
需要编写DynamicFactory工厂类,并提供非static方法返回Bean所示类对象
package com.itheima.factory;
……
public class DynamicFactory{
public UserDao getUserDao(){
return new UserDaoImpl();
}
}
*实例工厂方法变种(实现FactoryBean接口实例化,成MyBatis框架使用,后面详讲)
Spring核心配置文件配置如下:
<!-- 4.实现FactoryBean接口实例化 :实例工厂的变种,如集成MyBatis框架使用-->
<bean id="newDynamicFactory" class="com.itheima.factory.NewDynamicFactory"></bean>
编写一个类,实现org.springframework.beans.factory.FactoryBean接口并重写其方法
package com.itheima.factory;
import org.springframework.beans.factory.FactoryBean;
public class NewDynamicFactory implements FactoryBean<UserDao> {
@Override
public UserDao getObject() throws Exception {
UserDao userDao = new userDaoImpl();
return cat3;
}
@Override
public Class<?> getObjectType() {
return UserDao.class;
}
//从FactoryBean 中可以看出默认实单例模式
//可以修改为多例的,覆盖原来的方法
//public boolean isSingleton() {
// return false;
//}
}
3.5 Bean的手动注入(★非Spring原生提供支持★)
在web应用开发过程中,除了dao层对象,我们还需要创建service层对象,具体步骤如下:
①在Spring核心配置文件中配置UserServiceImpl、UserDaoImpl的Bean,将两者的创建权交由程序编写者交给Spring容器。
<bean id="userService" class="com.itheima.service.impl.UserServiceImpl"/>
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"/>
②编写UserServiceImpl、UserDaoImpl类,实现相应接口的方法。在持久层中要操作数据库等持久化内容。
public interface UserService {
public void save() {};
}
public class UserServiceImpl implements UserService {
@Override
public void save() {
//此处代码略,详见第③歩
//这里需要有一个userDao的对象,以便调用dao层操作数据库
//这种情况就是UserService依赖UserDao
}
}
public interface UserDao {
public void save() {};
}
public class UserDaoImpl implements UserDao{
@Override
public void save(){
//这里假装正在操作数据库等持久化内容
System.out.println("saving.....");
}
}
③因为要在UserServiceImpl中使用UserDaoImpl对象,所以在UserServiceImpl类的save()方法内部创建UserDaoImpl对象(当然这里对象的创建者是Spring容器),然后调用该对象的save方法。(重点)
public class UserServiceImpl implements UserService {
@Override
public void save() {
System.out.println("-------下面是重点代码-------");
ApplicationContext applicationContext = new
ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userDao = (UserDao) applicationContext.getBean("userDao");
userDao.save();
System.out.println("-------上面是重点代码-------");
}
}
④测试:l可以在maven的Test模块里面添加测试代码(或者在模拟的ctroller类中使用main方法测试)
ApplicationContext apc = new ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService = (UserService) apc.getBean("userService");
userService.save();
3.6 Bean依赖注入概念
在编写程序时,通过SPring的控制反转,我们把对象创建的工作交给了(反转给了)SPring容器,没有了硬编码。但是依赖关系并不会消除,Ioc只是降低了他们的依赖关系,eg:业务层仍需要调用Service层对象的方法。
既然service层、dao层的对象创建我们交给了Spring容器,那两层对象之间的关系是不是也可以交给Spring容器来管理呢?答案是肯定的。
简而言之,我们坐等框架(Spring容器)把持久层对象传入业务层,我们不需手动创建,直接使用即可。这种情况叫做依赖注入(Dependency Injection),在我们的案例里面就是:把持久层的对象注入到业务层。
依赖注入(Dependency Injection):它是 Spring 框架核心 IOC 的具体实现。
3.7 Bean依赖注入的三种方式
1)set方法注入:
在UserServiceImpl中添加userDao属性和setUserDao方法
public calss UserServiceImpl implements UserService{
private UserDao userDao;
public void setUserDao(UserDao userDao){
this.userDao = userDao;
}
public void save(){
userDao.save();
}
}
在Spring配置文件(容器)中为userService配置property子标签(注入属性)
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"></bean>
<bean id="userService" class="com.itheima.service.impl.UserServiceImpl">
<property name="userDao" ref="userDao"></property>
</bean>
**中有两个UserDao,但是代表的含义不一样,第一个是指userServiceImpl类中的属性名是UserDao,第二个userDao指的是xml配置文件中id=userDao的bean。
2)set方法:p命名空间注入
为了方便使用set方法进行注入,对xml编写格式进行了简化,其本质也是set方法注入。步骤如下:
首先,引入P命名空间:
xmlns:p="http://www.springframwwork.org/schema/p"
修改注入方式:
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl"></bean>
<bean id="userService" class="com.itheima.service.impl.UserServiceImpl" p:userDao-ref="userDao"></bean>
3)构造方法注入
在UserServiceImpl中添加有参构造,参数为userDao(为了健壮性,同时添加无参构造):
public class UserServiceImpl implements UserService{
private UserDao userDao;
//删除set方法,添加有参构造和无参构造
public UserServiceImpl(UserDao userDao){
this.userDao = userDao;
}
//因为很多场景下默认使用无参构造,为了程序的健壮性,添加无参构造
public UserServiceImpl(){}
//调用数据持久层方法操作数据库
public void save(){
userDao.save();
}
}
在Spring配置文件(容器)中为UserService配置constructor子标签(注入属性)
<bean id="userDao" class="com.itheima.userDao.impl.UserDaoImpl"></bean>
<bean id="userService" class="com.itheima.userServie.impl.UserServiceImpl">
<!-- name的值是构造方法的形参名称,ref的值是要赋值给形参的值,也就是容器中的Bean:UserDao-->
<constructor-arg name="userDao" ref="userDao"></constructor-arg>
</bean>
3.8 Bean依赖注入的数据类型
上面的操作过程中,注入的都是引用数据类型,通过ref属性关联被注入的数据。
除了引用数据类型之外,普通数据类型、集合都可以在容器中进行注入。
总而言之,可注入的类型分三种:
普通数据类型(基本数据类型 + 字符串)
引用数据类型(不含集合、字符串的引用数据类型)
集合数据类型(集合数据类型)
下面将以set方法注入方式为例,演示基本数据类型、集合数据类型的注入。
1)基本数据类型注入:
在UserDaoImpl中添加普通数据类型的成员变量和set方法
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 method running.....")
}
}
在Spring配置文件(容器)中,为UserDao配置property子标签(注入属性)
<bean id="userDao" class="com.ithiema.dao.impl.UserDaoImpl">
<property name="company" value="传智播客"></property>
<property name="age" value="18"></property>
</bean>
2)集合数据类型(List)注入
在UserDaoImpl中添加集合数据类型(List)的成员变量和set方法
public Class UserDaoImpl{
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....");
}
}
在Spring配置文件中为userDao设置子标签(注入属性)
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl">
<property name="strList">
<list>
<value>aaa</value>
<value>bbb</value>
<value>ccc</value>
</list>
</property>
</bean>
3)集合数据类型(List)注入
在UserDaoImpl中添加集合数据类型(List)的成员变量和set方法
public Class UserDaoImpl{
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....");
}
}
在Spring配置文件中为userDao设置子标签(注入属性)
<bean id="user1" class="com.itheima.domain.User">
<property name="name" value="张三"></property>
<property name="id" value="12"></property>
</bean>
<bean id="user2" class="com.itheima.domain.User">
<property name="name" value="李四"></property>
<property name="id" value="13"></property>
</bean>
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl">
<property name="userList">
<list>
<ref bean="user1"></ref>
<ref bean="user2"></ref>
<bean id="user3" class="com.itheima.domain.User">
<property name="name" value="王五"></property>
<property name="id" value="15"></property>
</bean>
</list>
</property>
</bean>
4)集合数据类型(Map)注入
在UserDaoImpl中添加集合数据类型(Map<String, User>)的成员变量和set方法
public Class UserDaoImpl{
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....");
}
}
在Spring配置文件中为userDao设置子标签(注入属性)
<bean id="user1" class="com.itheima.domain.User">
<property name="name" value="张三"></property>
<property name="id" value="12"></property>
</bean>
<bean id="user2" class="com.itheima.domain.User">
<property name="name" value="李四"></property>
<property name="id" value="13"></property>
</bean>
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl">
<property name="userMap">
<map>
<entry key="zhangsan" value-ref="user1"></entry>
<entry key="lisi" value-ref="user2"></entry>
</map>
</property>
</bean>
3)集合数据类型(Properties)注入
在UserDaoImpl中添加结合数据类型(Properties)的成员变量和set方法
public class UserDaoImpl implements UserDao {
private Properties pops;
public void setPops(Properties pops) {
this.pops = pops;
}
public void save() {
System.out.println(pops);
System.out.println("UserDao save method running....");
}
}
在Spring配置文件中为userDao设置子标签(注入属性)
<bean id="userDao" class="com.itheima.dao.impl.UserDaoImpl">
<property name="pops">
<props>
<prop key="p1">aaa</prop>
<prop key="p2">bbb</prop>
<prop key="p3">ccc</prop>
</props>
</property>
</bean>
3.9 Spring引入其他配置文件
在实际开发中,Spring配置文件内容会非常多,这就导致Spring配置繁杂而且容易出问题。
所以可以把Spring的配置文件按照业务需要、软件分层进行拆分,然后在主配置文件中引入其他配置文件。
<import resource="applicationContext-xxx.xml">
3.10 Spring配置文件总结
<bean>标签
id属性:在容器中Bean实例的唯一标识,不允许重复(可以没有)
class属性:要实例化的Bean的全限定名
scope属性:Bean的作用范围,常用是Singleton(默认)和prototype
<property>标签:属性注入
name属性:属性名称(通过截取set方法获得)
value属性:注入的普通属性值
ref属性:注入的容器中的bean对象的值
<list>标签
<value> 注入的普通属性值
<bean> 在list标签内部直接注入一个bean对象
<ref> 通过该标签的bean属性注入一个已有的bean对象
<map>标签
<entry> 一对键值对,key/key-erf为键,value/value-ref为值
<props>标签
<prop> key属性为key,标签内容为值
<constructor-arg>标签: 构造注入
name属性:构造方法的形参
ref属性:要注入的容器中的bean对象的值
value属性:要注入的普通属性值
<import>标签:导入其他的Spring的分文件
4 Spring常用API
4.1 ApplicationContext继承体系
ApplicationContext:接口,代表应用上下文(Spring容器),可以通过其实现类的对象获得Spring容器中的Bean对象。
4.2 ApplicationContext的实现类
1)ClassPathXmlApplicationContext
从类的根路径下加载xml配置文件,推荐使用这种(相对于第二种来说)
2)FileSystemXmlApplicationContext
从指定磁盘路径加载xml配置文件,配置文件可以在磁盘任意位置
3)AnnotationConfigApplicationContext
创建使用注解方式配置的Spring容器对象,用来解析识别注解。
4.3 getBean()方法使用
getBean()方法存在重载:
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有多个时,此方法会报错。
ApplicationContext applicationContext = new
ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService1 = (UserService) applicationContext.getBean("userService");
UserService userService2 = applicationContext.getBean(UserService.class);
getBean()方法存在重载:
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有多个时,此方法会报错。
ApplicationContext applicationContext = new
ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService1 = (UserService) applicationContext.getBean("userService");
UserService userService2 = applicationContext.getBean(UserService.class);