三大框架:
SSM: spring springMVC mybatis;
Spring框架学习day01
一、今日内容
1、spring的概述
spring是什么?
spring的两大核心?
spring的发展历程和优势
spring体系结构
2、Spring快速入门
3、IOC和DI
spring中基于XML的IOC的配置
4、相关api
二、Spring概述
2.1. 什么是Spring
Spring是分层的Java SE/EE应用 full-stack(全栈)轻量级开源框架,
以IOC(Inverse Of Control:控制反转)和AOP(Aspect Oriented Programming:面向切面编程)为内核;
提供了展现层Spring MVC和持久层Spring JDBC以及业务层事务管理等众多的企业级应用技术,还能整合开源世界众多著名的第三方框架和类库,逐渐成为使用最多的Java EE企业应用开源框架.
两大核心:
IOC: 控制反转: 将对象创建的权利交给spring; (不使用new );
AOP: 面向切面编程: 可以增强原有功能;
2.2. Spring的发展历程
1997年IBM提出了EJB的思想
1998年,SUN制定开发标准规范EJB1.0
1999年,EJB1.1发布
2001年,EJB2.0发布
2003年,EJB2.1发布
2006年,EJB3.0发布
Rod Johnson(spring之父) — spring生态
Expert One-to-One J2EE Design and Development(2002)
阐述了J2EE使用EJB开发设计的优点及解决方案
Expert One-to-One J2EE Development without EJB(2004)
阐述了J2EE开发不使用EJB的解决方式(Spring雏形)
2017年9月份发布了spring的最新版本spring 5.0通用版(GA)
---- 以前只有传统的大型公司才会开发商业web项目, 但是06年开始,有很多小型公司有需求,EJB对于他们来说太庞大了!(主要是从06年开始电商盈利,电商发展起来了)
特别说明:
spring5 版本是用 jdk8 编写的,所以要求我们的 jdk 版本是 8 及以上. 同时 tomcat 的版本要求 8.5 及以上.
2.3. Spring的优势
方便解耦,简化开发
通过Spring提供的IOC容器,可以将对象间的依赖关系交由Spring进行控制,避免硬编码所造成的过度程序耦合.用户也不必再为单例模式类、属性文件解析等这些很底层的需求编写代码,可以更专注于上层的应用.
AOP编程的支持
通过Spring的AOP功能,方便进行面向切面的编程,许多不容易用传统OOP实现的功能可以通过AOP轻松应付.
声明式事务的支持
可以将我们从单调烦闷的事务管理代码中解脱出来,通过声明式方式灵活的进行事务的管理,提高开发效率和质量.
方便程序的测试
可以用非容器依赖的编程方式进行几乎所有的测试工作,测试不再是昂贵的操作,而是随手可做的事情.
方便集成各种优秀框架(大管家)
Spring可以降低各种框架的使用难度,提供了对各种优秀框架(mybatis,springMVC、Struts、Hibernate、Hessian、Quartz等)的直接支持.
降低JavaEE API的使用难度
Spring对JavaEE API(如JDBC、JavaMail、远程调用等)进行了薄薄的封装层,使这些API的使用难度大为降低.
Java源码是经典学习范例
Spring的源代码设计精妙、结构清晰、匠心独用,处处体现着大师对Java设计模式灵活运用以及对Java技术的高深造诣.它的源代码无意是Java技术的最佳实践的范例.
2.4. Spring的体系结构
结论:
如果其他框架想要和spring进行一起使用,一般来说spring会去管理其他框架! (大管家)
三.Spring快速入门
①导入 Spring 开发的基本包坐标
②编写 Dao 接口和实现类
③创建 Spring 核心配置文件
④在 Spring 配置文件中配置 UserDaoImpl
⑤使用 Spring 的 API 获得 Bean 实例
Spring程序开发步骤
3.1.导入 Spring开发的基本包坐标
<groupId>com.itheima</groupId>
<artifactId>itheima_spring01_ioc</artifactId>
<version>1.0-SNAPSHOT</version>
<packaging>war</packaging>
<dependencies>
<dependency>
<groupId>org.springframework</groupId>
<artifactId>spring-context</artifactId>
<version>5.0.5.RELEASE</version>
</dependency>
<dependency>
<groupId>junit</groupId>
<artifactId>junit</artifactId>
<version>4.12</version>
</dependency>
</dependencies>
3.2.编写 Dao接口和实现类
public interface UserDao {
public void save();
}
public class UserDaoImpl implements UserDao {
@Override
public void save() {
System.out.println("UserDao save method running....");
}
}
3.3.创建 Spring核心配置文件
在(resources)创建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"
xmlns:p="http://www.springframework.org/schema/p"
xsi:schemaLocation="http://www.springframework.org/schema/beans http://www.springframework.org/schema/beans/spring-beans.xsd">
</beans>
3.4.在 Spring配置文件中配置UserDaoImpl
3.5.使用 Spring的API获得Bean 实例
@Test
public void test1(){
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userDao = (UserDao) applicationContext.getBean("userDao");
userDao.save();
}
基本环境搭建:
四、Spring配置:
4.1.Bean标签基本配置
用于配置对象交由Spring 来创建。
默认情况下它调用的是类中的无参构造函数,如果没有无参构造函数则不能创建成功。
基本属性:
id:Bean实例在Spring容器中的唯一标识
class:Bean的全限定名称
4.2.Bean标签范围配置
scope: 指对象的作用范围,取值如下:
取值范围 说明
singleton 默认值,单例的 – 整个项目运行过程中,只会有一个对象
prototype 多例的 – 每次获取,都会产生一个新的对象,有无数个
request WEB 项目中,Spring 创建一个 Bean 的对象,将对象存入到 request 域中
session WEB 项目中,Spring 创建一个 Bean 的对象,将对象存入到 session 域中
global session WEB 项目中,应用在 Portlet 环境,如果没有 Portlet 环境那么globalSession 相当于 session
1)当scope的取值为singleton时 – 单例
Bean的实例化个数:1个
Bean的实例化时机:当Spring核心文件被加载时,实例化配置的Bean实例
Bean的生命周期: (生命和容器一样长)
对象创建:当应用加载,创建容器时,对象就被创建了
对象运行:只要容器在,对象一直活着
对象销毁:当应用卸载,销毁容器时,对象就被销毁了
2)当scope的取值为prototype时 – 多例
Bean的实例化个数:多个
Bean的实例化时机:当调用getBean()方法时实例化Bean; (每次从容器获取对象,都会创建)
Bean的生命周期:
对象创建:当使用对象时,创建新的对象实例
对象运行:只要对象在使用中,就一直活着
对象销毁:当对象长时间不用时,被 Java 的垃圾回收器回收了; (GC不能人为控制销毁,是自动执行)
4.3.Bean生命周期配置
init-method: 指定类中的初始化方法名称
destroy-method: 指定类中销毁方法名称
4.4.Bean实例化三种方式(常用空参构造方式)
无参构造方法实例化
工厂静态方法实例化
工厂实例方法实例化
1) 使用无参构造方法实例化 (最常用)
它会根据默认无参构造方法来创建类对象,如果bean中没有默认无参构造函数,将会创建失败
2) 工厂静态方法实例化
工厂的静态方法返回Bean实例
常常用于,类不是我们自己写的, 没有提供空参构造时,使用!
3) 工厂实例方法实例化(了解)
工厂的非静态方法返回Bean实例
4.5.Bean的依赖注入入门-- 问题分析
①创建 UserService,UserService 内部在调用 UserDao的save() 方法
public class UserServiceImpl implements UserService {
@Override
public void save() {
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
UserDao userDao = (UserDao) applicationContext.getBean("userDao");
userDao.save();
}
}
②将 UserServiceImpl 的创建权交给 Spring
<bean id="userService" class="com.itheima.service.impl.UserServiceImpl"/>
③从 Spring 容器中获得 UserService 进行操作
public static void main(String[] args) {
ApplicationContext applicationContext = new ClassPathXmlApplicationContext("applicationContext.xml");
UserService userService = (UserService) applicationContext.getBean("userService");
userService.save();
}
---- 没有使用依赖注入时, 创建了多个容器对象, 这样其实是有问题的!
如何解决?
– 依赖注入
4.6.Bean的依赖注入概念–DI
依赖注入(Dependency Injection):它是 Spring 框架核心 IOC 的具体实现。
在编写程序时,通过控制反转,把对象的创建交给了 Spring,但是代码中不可能出现没有依赖的情况。
IOC 解耦只是降低他们的依赖关系,但不会消除。例如:业务层仍会调用持久层的方法。
那这种业务层和持久层的依赖关系,在使用 Spring 之后,就让 Spring 来维护了。
简单的说,就是坐等框架把持久层对象传入业务层,而不用我们自己去获取!
怎么将UserDao怎样注入到UserService内部呢?
方式: 两种:
Set属性注入
构造注入
到底什么是依赖注入?
简单来讲:通过容器给 对象的成员变量赋值;
4.7.Bean的依赖注入方式
① set方法注入(常用)
*在UserServiceImpl中添加setUserDao方法
*方式1: 配置Spring容器调用set方法进行注入(常用)
方式2: set方法: P命名空间注入 (了解)
P命名空间注入本质也是set方法注入,但比起上述的set方法注入更加方便,主要体现在配置文件中,如下:
Set注入优缺点:
优势: 创建对象时没有明确的限制,可以直接使用默认的无参构造函数 — 参数灵活
缺点: 如果某个成员必须有值,获取对象时set方法可能没有执行
② 构造方法注入
*创建有参构造
*配置Spring容器调用有参构造时进行注入
**补充:构造注入优缺点:
类中需要提供一个对应参数列表的构造函数。
涉及的标签: constructor-arg
优势:在获取bean对象时,数据注入是必须的,否则无法创建成功 — 能够要求创建对象时,强制给属性赋值;(比如创建数据库连接时)
缺点:改变了bean对象的实例化方式,使我们在创建对象时如果用不到某此数据同样也要提供。 – 不灵活
4.8.Bean的依赖注入的数据类型
注入数据的三种数据类型:
普通数据类型: string + 基本类型 – value
引用数据类型: service 、 dao – ref
集合数据类型: (了解)
其中引用数据类型,之前的操作都是对UserDao对象的引用进行注入的,下面将以set方法注入为例,演示普通数据类型和集合数据类型的注入。
1)引用数据类型的注入(掌握)
2)普通数据类型的注入(掌握)
3)集合数据类型的注入(了解)
a.集合数据类型(List)的注入
b.集合数据类型(List)的注入
c.集合数据类型( Map<String,User> )的注入
d.集合数据类型(Properties)的注入
4.9.引入其他配置文件(分模块开发)
实际开发中,Spring的配置内容非常多,这就导致Spring配置很繁杂且体积很大,所以,可以将部分配置拆解到其他配置文件中。
在Spring主配置文件通过import标签进行加载
五、Spring相关API
5.1.ApplicationContext的继承体系
applicationContext:接口类型,代表应用上下文,可以通过其实例获得 Spring 容器中的 Bean 对象
**补充:spring中工厂的类结构图
**补充:BeanFactory和ApplicationContext的区别
- BeanFactory是ApplicationContext的顶层接口
- BeanFactory是采用延迟加载策略(懒汉).只有真正使用getBean时才会实例化Bean.适用于多例模式
- ApplicationContext配置文件加载时就会立即初始化Bean(饿汉),适用于单例模式,实际开发时一般采用这种.
5.2.ApplicationContext的实现类–常用的三个实现类
1. ClassPathXmlApplicationContext:加载类路径下的配置文件,要求配置文件必须在类路径下(常用)
2. FileSystemXmlApplicationContext:加载磁盘任意路径下的配置文件(必须有访问权限)
3. AnnotationConfigApplicationContext:读取注解配置容器
5.3.getBean()方法使用
当参数的数据类型是字符串时,表示根据Bean的id从容器中获得Bean实例,返回是Object,需要强转。 — 根据id,获取的是唯一的
当参数的数据类型是Class类型时,表示根据类型从容器中匹配Bean实例,当容器中相同类型的Bean有多个时,则此方法会报错.
**补充
1.单例 和 多例:
单例: 在程序运行过程中, 多次获取,也只产生一个对象!
多例: 在程序运行过程中, 每次获取,都会产生一个新的对象!
区别:
个数:单例只创建一次, 多例创建多次;(一个程序运行过程中)
数据安全: 类成员变量:
单例存在数据泄露;
多例不存在数据安全问题;
局部变量,都不存在数据安全问题!
— > 解决方案 : 一般情况下,我们不会在单例对象中定义成员变量, 而是定义局部变量,可以避免这个问题;
执行效率:单例 > 多例 (多例每次创建需要消耗资源和时间);
单例常见有两种实现方式: 懒汉 、 饿汉; (区别:对象加载时机)
懒汉: 运行过程中,第一次调用时才创建(第一次使用时创建)
饿汉: 调用前,在程序加载时就创建(第一次直接使用)
2.程序的耦合和解耦
什么是程序的耦合
耦合就是程序间的依赖关系.
解耦就是降低程序间的依赖关系. (耦合只能降低,不能消除)