资源链接
文档链接
- Spring5.2.7 官方文档
- Spring5.1.3 中英文文档
教程链接
- 尚硅谷-Spring5框架2020最新版教程
程序下载
- Spring5.x 下载链接
Spring 框架基本概述
-
Spring
是一个开放源码的轻量级的JavaEE框架 -
Spring
可以解决企业应用开发的复杂性 -
Spring
有两个核心部分:IOC
和Aop
-
IOC
: 控制反转,把创建对象的过程交给Spring
管理 -
Aop
: 面向切面,不修改源代码的情况下,进行功能的增强
-
-
Spring
特点- 方便解耦,简化开发
-
Aop
编程支持 - 方便程序测试
- 方便和其他框架进行整合
- 方便进行事物操作
- 降低API开发难度
- 学习Spring5.x版本
入门案例
- 下载
Spring5
- 创建
Spring
工程 - 导入相关
Jar
包 - 创建普通类,在这个类创建普通方法
- 创建
Spring
配置文件
IOC:控制反转
什么是IOC
- 控制反转,把对象创建和对象之间的调用过程,交给
Spring
进行管理 - 使用
IOC
的目的:为了降低耦合度 - 做入门案例就是
IOC
实现
IOC
底层原理
- xml解析
- 工厂模式
- 反射
IOC
接口
IOC
思想基于IOC
容器完成,IOC
容器底层就是对象工厂-
Spring
提供IOC
容器实现两种方式(两个接口):- BeanFactory:
IOC
容器基本实现,是Spring内部的使用接口,不提供开发人员进行使用。- 加载配置文件的时候不会创建对象,在获取对象(使用)才去创建对象
- ApplicationContext: BeanFactory接口的子接口,提供更多更强大的功能,一般由开发人员使用。
- 加载配置文件时候就会把在配置文件对象进行创建
- BeanFactory:
ApplicationContext接口有实现类
IOC
的具体操作:Bean
管理
-
什么是
Bean
管理:-
Bean
管理指的是两个操作 - Spring创建对象
- Spring注入属性
-
-
Bean
管理操作有两种方式- 基于xml配置文件方式实现
- 基于注解方式实现
基于XML方式创建对象
- 在Spring配置文件中,使用bean标签,标签里面添加对应属性,就可以实现对象创建
- 在Bean标签中有很对属性
- id 属性:对象唯一标识
- class 属性:类全路径(包类路径)
- name 属性:较为早期的一个属性,同id属性,并可添加特殊字符
- 创建对象时候,默认也是执行无参数构造方法完成对象创建
基于XML注入属性
- DI:依赖注入,就是注入属性
- 第一种注入方式:使用set方法进行注入
- 第二种注入方式:使用有参数构造进行注入
第一种注入方式:使用set方法进行注入
- 创建类,定义属性和对应的set方法
xml 注入其他类型属性
-
字面量
- null 值
- 属性值包含特殊符号
-
属性注入:外部bean
- 创建两个类service类和dao类
- 在service调用dao里面的方法
-
属性注入:内部bean
- 一对多关系:部门和员工
- 实体类之间表示一对多的关系
- 在配置文件中进行配置
属性注入:级联赋值
xml 注入集合类型属性
- 注入数组类型属性
- 注入List集合类型属性
- 注入Map集合类型属性
- 在集合里面设置对象类型值
- 把集合注入部分提取出来
- 在Spring的配置文件中引入名称空间 util
- 使用 util 标签完成list集合注入提取
FactoryBean
- Spring里有两种类型的bean,一种普通bean,另外一种工厂bean:FactoryBean
- 普通bean:在配置文件中定义的类型就是返回的类型
- 工厂bean:在配置文件中定义的bean的类型可以和返回类型不一样
- 第一步:创建一个类,让这个类作为工厂bean,实现类接口 FactoryBean
- 第二步:实现接口里的方法,返回类型为
bean的作用域
- 在Spring里面,设置创建的Bean实例是单实例对象还是多实例
- 单实例:只创建一次,可以使用查看地址验证
- 多示例:每次都重新创建
- 在Spring里面,默认情况下,创建单实例bean
- 如何设置多实例
- 在Spring bean标签里,有一个属性(Scope)可以设置
- scope属性值:
- 默认值:singleton:表示是单实例对象
- prototype:表示是多实例对象
- request:
- session:
- singleton和prototype区别
- singleton:表示是单实例对象;prototype:表示是多实例对象
- singleton:加载配置文件时就会创建实例对象
- prototype:使用getBean方法时,才创建新的对象
bean的生命周期
- 生命周期
- 从对象的创建到对象销毁的过程
- bean的生命周期
- 通过构造器创建bean实例(无参数构造)
- 为bean的属性值和对其他bean引用(调用set方法)
- 调用bean的初始化的方法
- bean可以使用了(对象获取到了)
- 当容器关闭时候,调用bean的销毁方法(需要进行配置销毁的方法)
- 演示bean的生命周期
- bean的后置处理器,bean生命周期一共有七步
- 通过构造器创建bean实例(无参数构造)
- 为bean的属性值和对其他bean引用(调用set方法)
- 把bean实例传递bean后置处理器方法
- 调用bean的初始化的方法
- 把bean实例传递bean后置处理器方法
- bean可以使用了(对象获取到了)
- 当容器关闭时候,调用bean的销毁方法(需要进行配置销毁的方法)
- 添加后置处理器效果
问题1:后置处理器有什么作用
问题2:bean生命周期每一步的实现细节
xml自动装配
- 什么是自动装配
- 指定装配规则(属性名称或者属性类型),Spring自动将匹配的属性值进行注入
- 演示自动装配过程
- 根据属性名称实现自动装配
- 根据属性类型进行自动注入
bean 标签属性autowire,装配自动装配
autowire属性常用两个值:
1. byName 根据属性名称注入,注入值bean的id值和类属性名称一样
2. byType根据属性类型注入: 相同类型的bean只能有一个,否则会找不到
外部属性文件
- 配置数据库信息
- 配置德鲁伊连接池
- 引入德鲁伊连接池依赖jar包
- 引入外部的属性文件配置数据库的连接池
- 创建外部属性文件,properities格式文件,写数据库信息
- 引入 context 名称空间
- spring配置文件使用标签引入外部属性文件
基于注解方式注入属性
- 什么是注解
- 代码特殊的标记 格式:@注解名称(属性名称=属性值, 属性名称=属性值...)
- 使用注解:注解作用在类上面、方法上面、属性上面
- 使用注解目的:简化xml配置
- spring针对bean管理创建对象提供注解
- @Component
- @Service
- @Controller
- @Repository
- * 上面四个注解功能是一样的都可以用来创建bean实例
- 注解使用
- 第一步:引入依赖-aop
- 第二步:开启组件扫描
- 第三步:创建类,在类上添加创建对象注解
- 基于注解方式实现属性注入
- @AutoWired: 更具属性类型进行自动装配
- 第一步:吧service和dao对象创建,在service和dao类添加创建对象注解
- 第二步:在service里面注入dao对象,在service类添加dao类型属性,在属性上面使用注解
- @Qualifier: 更具属性名称进行注入
- @Qualifier注解的使用要和@AutoWired一起使用
- @Resource: 可以根据类型注入,可以根据名称注入
- @Value: 注入普通属性
- @AutoWired: 更具属性类型进行自动装配
- 纯注解开发
- 创建配置类,替代xml配置文件
- 编写测试类
AOP
- 什么是AOP:面向切面编程。利用AOP可以对业务逻辑的各个部分进行隔离,从而使得业务逻辑各个部分之间的耦合度降低,提高程序的可重用性,同时提高了开发效率。
- 通俗讲解:不通过修改源代码,在主干中添加新功能
-
使用例子说明AOP
AOP底层原理
- AOP底层使用动态代理
- 有两种情况的动态代理
- 第一种:有接口的情况,使用JDK动态代理
- 创建接口实现类代理对象,增强类的方法
- 第二种:没有接口情况,使用CGLIB动态代理
- 创建子类的代理对象,增强类的方法
- 第一种:有接口的情况,使用JDK动态代理
- 有两种情况的动态代理
JDK动态代理
- 使用JDK动态代理,
- 调用newProxyInstance方法
public static Object newProxyInstance( ClassLoader loader, Class>[] interfaces, InvocationHandler h ) Returns a proxy instance for the specified interfaces that dispatches method invocations to the specified invocation handler. 1. loader - the class loader to define the proxy class | 类加载器 2. interfaces - the list of interfaces for the proxy class to implement | 增强方法所在类,这个类实现的接口,支持多个接口 3. h - the invocation handler to dispatch method invocations to | 实现这个接口 InvocationHandler,创建代理对象,写增强的方法
- 调用newProxyInstance方法
- 实现JDK动态代理代码
- 创建接口,实现方法
AOP 术语
- 连接点:类里面哪些方法可以被增强,这些方法称为连接点
- 切入点:实际被真正增强的方法,称为切入点
- 通知(增强)
- 实际增强的逻辑部分称为通知(增强)
- 通知有多种类型
- 前置通知
- 后置通知
- 环绕通知
- 异常通知
- 最终通知
- 切面:是动作
- 把通知应用到切入点的过程
AOP操作
- Spring框架中一般基于AspectJ实现AOP操作
- 什么是AspectJ:
- AspectJ不是Spring组成部分,独立AOP框架,一般把AspectJ和Spring框架一起使用,进行AOP操作。
- 什么是AspectJ:
- 基于AspectJ实现AOP操作
- 基于xml配置文件实现
- 基于注解方式实现
- 在项目工程里面引入AOP相关依赖
- 切入点表达式
- 切入点表达式的作用:知道对哪个类里面的哪个方法进行增强
- 语法结构:
execution([权限修饰符][返回类型][类全路径][方法名称]([参数列表]))
- 举例1:对
com.atguigu.dao.DaoBook
类里面的add
进行增强:execution(* com.atguigu.dao.DaoBook.add(..))
- 举例2:对
com.atguigu.dao.DaoBook
类里面的所有方法进行增强:execution(* com.atguigu.dao.DaoBook.*(..))
- 举例3:对
com.atguigu.dao
包里面所有类,类里面的所有方法进行增强execution(* com.atguigu.dao.*.*(..))
AspectJ注解
-
创建类,在类里面定义方法
// 被增强的类 public class AopUser { public void add(){ System.out.println("AopUser Add......."); } }
-
创建增强类(编写增强的逻辑)
- 在增强的类中创建方法
// 增强的类 public class AopUserProxy { // 前置通知 public void before() { System.out.println("AopUserProxy Before........"); } }
- 在增强的类中创建方法
-
进行通知的配置
- 在spring配置文件中,开启注解扫描
- 使用注解创建User和UserProxy对象
- 在增强类上面添加注解@Aspect
- 在spring配置文件中开启代理对象
- 配置不同类型的通知
- 在增强类的里面,在作为通知方法上面添加通知类型注解,使用切入点表达式配置
// 增强的类 @Component @Aspect //生成代理对象 public class AopUserProxy { // 前置通知 // @Before 注解表示作为前置通知,表示给哪个类里面的哪个方法进行增强 @Before(value = "execution(* thx.study.spring.aop.AopUser.add(..))") public void before() { System.out.println("AopUserProxy Before........"); } // 最终通知,有没有异常都会执行 // 方法之后执行 @After(value = "execution(* thx.study.spring.aop.AopUser.add(..))") public void after() { System.out.println("AopUserProxy After........"); } // 返回通知(最终通知),存在异常就不执行 // 返回之后执行 @AfterReturning(value = "execution(* thx.study.spring.aop.AopUser.add(..))") public void afterReturning() { System.out.println("AopUserProxy AfterReturning........"); } // 异常通知 @AfterThrowing(value = "execution(* thx.study.spring.aop.AopUser.add(..))") public void afterThrowing() { System.out.println("AopUserProxy AfterThrowing........"); } // 环绕通知 @Around(value = "execution(* thx.study.spring.aop.AopUser.add(..))") public void around(ProceedingJoinPoint proceedingJoinPoint) throws Throwable { System.out.println("AopUserProxy Around Before........"); // 被增强的方法执行 proceedingJoinPoint.proceed(); System.out.println("AopUserProxy Around After........"); } }
// testAop @Test public void testAop() { // 1. 加载配置文件 ApplicationContext context = new ClassPathXmlApplicationContext("beanAop.xml"); // 2. 获取配置创建的对象 AopUser aopUser = context.getBean("aopUser", AopUser.class); System.out.println(aopUser); aopUser.add(); }
-
公共切入点抽取
- 使用
@Pointcut
注解
// 抽取相同的切入点 @Pointcut(value = "execution(* thx.study.spring.aop.AopUser.add(..))") public void pointDemo() { } // 前置通知 // @Before 注解表示作为前置通知,表示给哪个类里面的哪个方法进行增强 @Before(value = "pointDemo()") public void before() { System.out.println("AopUserProxy Before........"); } // 最终通知,有没有异常都会执行 // 方法之后执行 @After(value = "pointDemo()") public void after() { System.out.println("AopUserProxy After........"); }
- 使用
-
有多个增强类对同一个方法进行增强,设置增强类的优先级
- 在增强类上面添加注解
@Order(数字类型值)
,数字类性值越小优先级越高
@Component @Aspect //生成代理对象 @Order(1) public class AopUserProxy
- 在增强类上面添加注解
-
完全使用注解开发
- 创建配置类,不需要创建xml配置文件
@Configuration @ComponentScan(basePackages = "thx.study.spring.aop") @EnableAspectJAutoProxy(proxyTargetClass = true) public class ConfigAop { }
AspectJ配置文件(很少用)
- 创建两个类,增强类和被增强类,创建方法
- 在spring配置文件中创建两个类对象
- 在spring配置文件中配置切入点
JdbcTemplate
- 什么是JdbcTemplate
- Spring框架对JDBC进行封装,使用JdbcTemplate方便实现数据库的操作
- 准备工作
- 引入依赖
- 在spring配置文件中配置数据库连接池
- 配置JdbcTemplate对象,注入DataSource
- 创建service类,创建dao类,在dao注入jdbcTemplate对象
JdbcTemplate操作数据库
- 对应数据库创建实体类
- 编写service和dao
- 在dao进行数据库添加操作
- 调用JdbcTemplate的update方法实现数据库添加操作
- 第一个参数:sql语句
- 第二个参数:可变参数,设置sql语句值
@Repository public class JdbcBookDaoImpl implements JdbcBookDao{ @Autowired private JdbcTemplate jdbcTemplate; // 添加的方法 @Override public void add(JdbcBook jdbcBook) { // 1. 创建 sql 语句 String sql = "insert into t_book values(?,?,?)"; // 2. 调用方法实现 Object[] args = {jdbcBook.getUserId(), jdbcBook.getUserName(), jdbcBook.getUserState()}; int update = jdbcTemplate.update( sql, args ); System.out.println(update); } }